pax_global_header00006660000000000000000000000064141574043120014513gustar00rootroot0000000000000052 comment=82eb53208e48967198444aca29879a0a7cafd199 pineapple-pictures-0.5.3/000077500000000000000000000000001415740431200153315ustar00rootroot00000000000000pineapple-pictures-0.5.3/.github/000077500000000000000000000000001415740431200166715ustar00rootroot00000000000000pineapple-pictures-0.5.3/.github/workflows/000077500000000000000000000000001415740431200207265ustar00rootroot00000000000000pineapple-pictures-0.5.3/.github/workflows/macos.yml000066400000000000000000000004561415740431200225600ustar00rootroot00000000000000name: macOS CI on: [push, pull_request] jobs: build: runs-on: macos-latest steps: - uses: actions/checkout@v1 - name: Install Qt uses: jurplel/install-qt-action@v2 with: version: '5.15.2' - name: Run a qt project run: | cmake ./ make pineapple-pictures-0.5.3/.github/workflows/ubuntu.yml000066400000000000000000000011451415740431200227740ustar00rootroot00000000000000name: Ubuntu 20.04 CI on: [push, pull_request] jobs: build: runs-on: ubuntu-20.04 steps: - uses: actions/checkout@v1 - name: Get build dept. run: | sudo apt update sudo apt install cmake qtbase5-dev libqt5svg5-dev qttools5-dev libexiv2-dev - name: Build it run: | mkdir build cd build cmake ../ make cpack -G DEB - name: Try install it run: | cd build sudo apt install ./*.deb - uses: actions/upload-artifact@v2 with: name: ubuntu-20.04-deb-package path: build/*.deb pineapple-pictures-0.5.3/.gitignore000066400000000000000000000001261415740431200173200ustar00rootroot00000000000000# User files *.user *.user.* # Translation files *.qm # Generic Build Dir [Bb]uild/ pineapple-pictures-0.5.3/CMakeLists.txt000066400000000000000000000155311415740431200200760ustar00rootroot00000000000000project (pineapple-pictures) cmake_minimum_required (VERSION 3.9.5) list(APPEND CMAKE_MODULE_PATH ${CMAKE_CURRENT_LIST_DIR}/cmake) include (GNUInstallDirs) include (FeatureSummary) set (CMAKE_AUTOMOC ON) set (CMAKE_AUTORCC ON) set (QT_MINIMUM_VERSION "5.10") option (EXIV2_METADATA_SUPPORT "Better image metadata support via libexiv2" ON) find_package(Qt5 ${QT_MINIMUM_VERSION} CONFIG REQUIRED Widgets Svg LinguistTools) if (EXIV2_METADATA_SUPPORT) find_package(LibExiv2) set_package_properties(LibExiv2 PROPERTIES URL "https://www.exiv2.org" DESCRIPTION "image metadata support" TYPE OPTIONAL PURPOSE "Bring better image metadata support" ) endif () #LibExiv2_FOUND set (PPIC_CPP_FILES app/main.cpp app/framelesswindow.cpp app/mainwindow.cpp app/actionmanager.cpp app/graphicsview.cpp app/graphicsscene.cpp app/bottombuttongroup.cpp app/navigatorview.cpp app/opacityhelper.cpp app/toolbutton.cpp app/settings.cpp app/settingsdialog.cpp app/aboutdialog.cpp app/metadatamodel.cpp app/metadatadialog.cpp app/exiv2wrapper.cpp app/playlistmanager.cpp ) set (PPIC_HEADER_FILES app/framelesswindow.h app/mainwindow.h app/actionmanager.h app/graphicsview.h app/graphicsscene.h app/bottombuttongroup.h app/navigatorview.h app/opacityhelper.h app/toolbutton.h app/settings.h app/settingsdialog.h app/aboutdialog.h app/metadatamodel.h app/metadatadialog.h app/exiv2wrapper.h app/playlistmanager.h ) set (PPIC_QRC_FILES assets/resources.qrc ) set (PPIC_RC_FILES # yeah, it's empty. ) set (EXE_NAME ppic) # Translation file (GLOB PPIC_TS_FILES app/translations/*.ts) set (PPIC_CPP_FILES_FOR_I18N ${PPIC_CPP_FILES}) qt5_create_translation(PPIC_QM_FILES ${PPIC_CPP_FILES_FOR_I18N} ${PPIC_TS_FILES}) if (WIN32) list(APPEND PPIC_RC_FILES assets/pineapple-pictures.rc) endif () add_executable (${EXE_NAME} ${PPIC_HEADER_FILES} ${PPIC_CPP_FILES} ${PPIC_QRC_FILES} ${PPIC_RC_FILES} ${PPIC_QM_FILES} ) target_link_libraries (${EXE_NAME} Qt5::Widgets Qt5::Svg) if (LibExiv2_FOUND) message(INFO ${LibExiv2_INCLUDE_DIRS}) target_include_directories(${EXE_NAME} PRIVATE ${LibExiv2_INCLUDE_DIRS} ) target_link_libraries (${EXE_NAME} LibExiv2::LibExiv2 ) target_compile_definitions(${EXE_NAME} PRIVATE HAVE_EXIV2_VERSION="${LibExiv2_VERSION}" ) endif () # Extra build settings if (WIN32) set_property ( TARGET ${EXE_NAME} PROPERTY WIN32_EXECUTABLE true ) target_compile_definitions(${EXE_NAME} PRIVATE FLAG_PORTABLE_MODE_SUPPORT=1 ) endif () # Helper macros for parsing and setting project version from `git describe --long` result macro (ppic_set_version_via_describe _describe_long) string ( REGEX REPLACE "^([0-9a-z.]*)-[0-9]+-g[0-9a-f]*$" "\\1" _tag_parts "${_describe_long}" ) list (GET _tag_parts 0 _matched_tag_version) if ("${_matched_tag_version}" MATCHES "^[0-9]+\\.[0-9]+\\.[0-9]+$") string ( REGEX REPLACE "^([0-9]+)\\.([0-9]+)\\.([0-9]+).*$" "\\1;\\2;\\3" _ver_parts "${_matched_tag_version}" ) list (GET _ver_parts 0 CPACK_PACKAGE_VERSION_MAJOR) list (GET _ver_parts 1 CPACK_PACKAGE_VERSION_MINOR) list (GET _ver_parts 2 CPACK_PACKAGE_VERSION_PATCH) endif () endmacro () # Version setup if (EXISTS "${CMAKE_SOURCE_DIR}/.git") find_package(Git) set_package_properties(Git PROPERTIES TYPE OPTIONAL PURPOSE "Determine exact build version.") if (GIT_FOUND) execute_process ( COMMAND ${GIT_EXECUTABLE} describe --tags --always --long WORKING_DIRECTORY ${CMAKE_SOURCE_DIR} OUTPUT_VARIABLE _git_describe_long ) string (REGEX REPLACE "\n" "" _git_describe_long "${_git_describe_long}") ppic_set_version_via_describe(${_git_describe_long}) target_compile_definitions(${EXE_NAME} PRIVATE GIT_DESCRIBE_VERSION_STRING="${_git_describe_long}" ) endif () endif () # Helper macros for install settings macro (ppic_convert_to_relative_path _var) # Make sure _var is a relative path if (IS_ABSOLUTE "${${_var}}") file (RELATIVE_PATH ${_var} "${CMAKE_INSTALL_PREFIX}" "${${_var}}") endif () endmacro () # Install settings if (WIN32) # FIXME: try to avoid install to a "bin" subfolder under windows... # when fixed, don't forget to update the CI config file... set (BIN_INSTALL_DIR "") # seems useless, don't know why... elseif (UNIX) if (CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT) set(CMAKE_INSTALL_PREFIX /usr) endif () set (BIN_INSTALL_DIR "${CMAKE_INSTALL_BINDIR}") # relative, usually "bin" ppic_convert_to_relative_path(BIN_INSTALL_DIR) set (LIB_INSTALL_DIR "${CMAKE_INSTALL_LIBDIR}") # "lib" or "lib64" ppic_convert_to_relative_path(LIB_INSTALL_DIR) # install icon install ( FILES assets/icons/app-icon.svg DESTINATION "${CMAKE_INSTALL_DATADIR}/icons/hicolor/48x48/apps" RENAME pineapple-pictures.svg ) # install shortcut install ( FILES dist/net.blumia.pineapple-pictures.desktop DESTINATION "${CMAKE_INSTALL_DATAROOTDIR}/applications" ) # install app metadata file for appstream (and some other stuff using this metadata like snapcraft) install ( FILES dist/appstream/net.blumia.pineapple-pictures.metainfo.xml DESTINATION "${CMAKE_INSTALL_DATAROOTDIR}/metainfo" ) endif() set (INSTALL_TARGETS_DEFAULT_ARGS RUNTIME DESTINATION ${BIN_INSTALL_DIR} LIBRARY DESTINATION ${LIB_INSTALL_DIR} ARCHIVE DESTINATION ${LIB_INSTALL_DIR} COMPONENT Devel ) install ( TARGETS ${EXE_NAME} ${INSTALL_TARGETS_DEFAULT_ARGS} ) if (WIN32) set (QM_FILE_INSTALL_DIR "${CMAKE_INSTALL_BINDIR}/translations") else () set (QM_FILE_INSTALL_DIR "${CMAKE_INSTALL_FULL_DATADIR}/pineapple-pictures/translations") target_compile_definitions(${EXE_NAME} PRIVATE QM_FILE_INSTALL_DIR=${QM_FILE_INSTALL_DIR} ) endif () install ( FILES ${PPIC_QM_FILES} DESTINATION ${QM_FILE_INSTALL_DIR} ) feature_summary(WHAT ALL INCLUDE_QUIET_PACKAGES FATAL_ON_MISSING_REQUIRED_PACKAGES) # CPACK: General Settings set (CPACK_GENERATOR "TBZ2") set (CPACK_PACKAGE_NAME "pineapple-pictures") set (CPACK_PACKAGE_DESCRIPTION_SUMMARY "Yet another image viewer") set (CPACK_PACKAGE_VENDOR "Gary Wang") set (CPACK_PACKAGE_CONTACT "https://github.com/BLumia/pineapple-pictures/issues/") if (WIN32) # ... elseif (APPLE) # ... elseif (UNIX) set (CPACK_SYSTEM_NAME "${CMAKE_SYSTEM_NAME}-${CMAKE_SYSTEM_PROCESSOR}") set (CPACK_DEBIAN_PACKAGE_SHILIBDEPS ON) set (CPACK_DEBIAN_PACKAGE_RECOMMENDS "kimageformat-plugins") endif() include(CPack) pineapple-pictures-0.5.3/LICENSE000066400000000000000000000020741415740431200163410ustar00rootroot00000000000000MIT License Copyright (c) 2020 BLumia 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. pineapple-pictures-0.5.3/README.md000066400000000000000000000116031415740431200166110ustar00rootroot00000000000000Yet another image viewer. |CI|Build Status| |---|---| |Windows Build|[![Windows build status](https://ci.appveyor.com/api/projects/status/dbd8clww3cit6oa0/branch/master?svg=true)](https://ci.appveyor.com/project/BLumia/pineapplepictures/branch/master)| |macOS Build|![macOS CI](https://github.com/BLumia/pineapple-pictures/workflows/macOS%20CI/badge.svg)| |Ubuntu 20.04 Build|![Ubuntu 20.04 CI](https://github.com/BLumia/pineapple-pictures/workflows/Ubuntu%2020.04%20CI/badge.svg)| ![Pineapple Pictures - Main Window](https://repository-images.githubusercontent.com/211888654/e8697600-e370-11eb-9b2a-b71e05262954) ## Summary Pineapple Pictures is a lightweight image viewer that allows you view JPEG, PNG, GIF, SVG, PSD, KRA, XCF, TGA, AVIF and some other frequently used image formats files quickly and easily, and also provide a Stay-on-Top window setting that allows you pin the window so you can use it to pin a reference image at the top and then you can work with other software. ## Get it! - [GitHub Release Page](https://github.com/BLumia/pineapple-pictures/releases) - [SourceForge](https://sourceforge.net/projects/pineapple-pictures/) - Archlinux AUR: [pineapple-pictures](https://aur.archlinux.org/packages/pineapple-pictures/) | [pineapple-pictures-git](https://aur.archlinux.org/packages/pineapple-pictures-git/) - Debian (since bullseye) or Ubuntu (since 21.04): `sudo apt install pineapple-pictures` - [Itch.io Store](https://blumia.itch.io/pineapple-pictures) ## Help Translation! [Translate this project on Weblate!](https://hosted.weblate.org/projects/pineapple-pictures/) ## Build it manually: Current state, we need: - `cmake`: as the build system. - `qt5` with `qt5-svg` and `qt5-tools`: since the app is using Qt. - `libexiv2`: able to display more image metadata. (optional, but recommended) Then we can build it with any proper c++ compiler like g++ or msvc. Building it just requires normal cmake building steps: ``` bash $ mkdir build && cd build $ cmake .. $ cmake --build . # or simply using `make` if you are using Makefile as the cmake generator. ``` After that, a `ppic` executable file will be available to use. You can also optionally install it by using the target `install` (or simply `make install` in case you are using Makefile). After the build process, you can also use `cpack` to make a package. The project will try to build with `exiv2` when it's available at build time, if you would like to build the project without `exiv2`, pass `-DEXIV2_METADATA_SUPPORT=OFF` to `cmake`. The project will also not use `exiv2` if it's not found, the `EXIV2_METADATA_SUPPORT` option can be useful if you have `exiv2` but specifically don't want to use it. Image formats supports rely on Qt's imageformats plugins, just get the plugins you need from your distro's package manager will be fine. For Windows user, you may need build and install the imageformats plugin manually, read the content below. ### Linux Just normal build process as other program will be fine. Nothing special ;) For Archlinux there are also a [PKGBUILD](https://aur.archlinux.org/cgit/aur.git/tree/PKGBUILD?h=pineapple-pictures-git) you can use. For packaging to debian-based distro, the `CMakeLists.txt` provides some cpack configurations for generating a `.deb` package. After the build process, use `cpack -G DEB` to generate the package. You can also take `.github/workflows/ubuntu.yml` as a reference. For this project, `DEB` is the only supported cpack generator in current state, feel free to submit a PR if you like improving `cpack` support for this project. ### Windows The normal build steps for Linux is also applied to Windows, but since Windows doesn't have a decent package manager, so if you need any other image formats support other than the supported formats which Qt provided, you need to get and build these imageformats plugins manually and vendor it. It's optional and can be skipped if you don't need extra image formats support. For the Windows binary I provided, kimageformats plugin is used (for formats like kra, xcf, psd and etc.). You can take `appveyor.yml` as a reference to learn what I did when building the Windows binary. [KDE Craft](https://community.kde.org/Craft) environment also can be used to build and package this program. I did also created a blueprint for building this project that you can found it at [here](https://github.com/BearKidsTeam/craft-shmooprint-bkt). It's not the way I used to create the release binary, but still worth trying. ### macOS I don't have a mac, so no support at all. There is also a GitHub Action (see `.github/workflows/macos.yml`) running macOS build though so at least it can build. Feel free to submit a PR if you would like to give some love to the macOS build ;P ## License Pineapple Pictures as a whole is licensed under MIT license. Individual files may have a different, but compatible license. pineapple-pictures-0.5.3/README.zh_CN.md000066400000000000000000000126651415740431200176220ustar00rootroot00000000000000简单轻量的跨平台看图工具。 |CI|构建状态| |---|---| |Windows Build|[![Windows build status](https://ci.appveyor.com/api/projects/status/dbd8clww3cit6oa0/branch/master?svg=true)](https://ci.appveyor.com/project/BLumia/pineapplepictures/branch/master)| |macOS Build|![macOS CI](https://github.com/BLumia/pineapple-pictures/workflows/macOS%20CI/badge.svg)| |Ubuntu 20.04 Build|![Ubuntu 20.04 CI](https://github.com/BLumia/pineapple-pictures/workflows/Ubuntu%2020.04%20CI/badge.svg)| ![Pineapple Pictures - Main Window](https://repository-images.githubusercontent.com/211888654/e8697600-e370-11eb-9b2a-b71e05262954) ## 简介 菠萝看图是一个轻量图像查看器,允许你简单快捷的查看 JPEG, PNG, GIF, SVG, PSD, KRA, XCF, TGA, AVIF 等常用格式的图像文件,并提供了置顶窗口的选项以便你在使用其它软件时也可以将参考图片固定在顶端。 ## 立即获取! - [GitHub Release 页面](https://github.com/BLumia/pineapple-pictures/releases) | [gitee 发布页面](https://gitee.com/blumia/pineapple-pictures/releases) - [SourceForge](https://sourceforge.net/projects/pineapple-pictures/) - Archlinux AUR: [pineapple-pictures](https://aur.archlinux.org/packages/pineapple-pictures/) | [pineapple-pictures-git](https://aur.archlinux.org/packages/pineapple-pictures-git/) - Debian (自 bullseye 起) 或 Ubuntu (自 21.04 起): `sudo apt install pineapple-pictures` - [Itch.io 商店](https://blumia.itch.io/pineapple-pictures) ## 帮助翻译! [在 Weblate 上帮助此项目翻译到更多语言!](https://hosted.weblate.org/projects/pineapple-pictures/) ## 手动构建步骤: 当前状态,我们需要先确保如下依赖可用: - `cmake`: 我们所使用的构建系统 - 包含 `qt5-svg` 与 `qt5-tools` 组件的 `qt5`: 此应用基于 Qt - `libexiv2`: 用以获取和显示更多的图像元信息(可选,推荐) 然后我们就可以使用任何常规的 c++ 编译器如 g++ 或 msvc 来进行构建了 构建过程就是常规的 CMake 应用构建过程: ``` bash $ mkdir build && cd build $ cmake .. $ cmake --build . # 如果你使用 Makefile 作为 CMake 生成器,也可以直接简单的使用 `make` ``` 完毕后,一个名为 `ppic` 的可执行程序即会被生成以供使用。您也可以选择通过使用 CMake 生成的 `install` 目标继续将其安装到您的设备上(假设您使用 Makefile,即可执行 `make install` 来进行安装)。构建步骤完毕后,您也可以使用 `cpack` 来对应用程序进行打包。 当 `exiv2` 在构建时可用时,此项目将尝试使用其进行构建,若您不希望使用 `exiv2`,请传递 `-DEXIV2_METADATA_SUPPORT=OFF` 参数给 `cmake`。此项目在找不到 `exiv2` 时并不会使用 `exiv2`,`EXIV2_METADATA_SUPPORT` 选项可供尽管存在可用的 `exiv2` 但您明确不希望启用其支持时使用。 此应用的图片格式支持依赖于 Qt 的 imageformats 插件,直接从您所用的发行版获取对应的图像格式插件即可。对于 Windows 用户,您可能需要手动构建和使用图像格式插件。下方给出了进一步的说明。 ### Linux 常规的构建步骤即可完成构建,不需要额外的处理步骤 ;) 对于 Archlinux 发行版的用户,这里还有一个 [PKGBUILD](https://aur.archlinux.org/cgit/aur.git/tree/PKGBUILD?h=pineapple-pictures-git) 可供使用和参考。 对于在基于 debian 的发行版中进行打包的需求, `CMakeLists.txt` 已经提供了一些基本的 cpack 配置以便生成一个有效的 `.deb` 软件包。在构建步骤完毕后,使用 `cpack -G DEB` 即可生成 DEB 软件包。您也可以参考 `.github/workflows/ubuntu.yml` 来查看当前正在使用的 CI 配置是如何进行打包的。 目前,`DEB` 是当前唯一受到直接支持的 cpack 生成目标。若希望为此项目添加其它的 cpack 目标支持,欢迎发起合并请求。 ### Windows 上述的构建步骤在 Windows 中也适用,但由于 Windows 中不具备类如大多 Linux 发行版中所提供的方便的软件包管理机制,故如果您需要任何 Qt 官方支持之外的图像格式例如 psd,xcf,kra 等格式的支持,你就可能需要自行获取并构建对应的 imageformats 插件,并在您最终生成的可执行文件中一并提供这些插件。若您不需要这些额外的图像格式支持,这个步骤也可以直接跳过。 我们所提供的预编译好的 Windows 程序包含了 kimageformats 插件来提供额外(kra, xcf, psd 等)格式的支持。您可以参考 `appveyor.yml` 来查看我们是如何构建并打包 Windows 可执行程序的。 [KDE Craft](https://community.kde.org/Craft) 环境也可以被用来构建此应用程序。我也创建了一个蓝图来进行此项目的构建和打包,可参见[这里](https://github.com/BearKidsTeam/craft-shmooprint-bkt)。尽管这不是我用于构建发布二进制所使用的方案,但仍值得一试。 ### macOS 由于我没有 mac 设备,故 macOS 暂时不受任何支持。不过我们目前有一个 GitHub Action 来执行 macOS 环境下的构建(见 `.github/workflows/macos.yml`)所以至少 macOS 下是可以顺利进行构建的。如果您想完善对 macOS 的支持,也欢迎您创建合并请求 ;P ## 许可协议 菠萝看图整体使用 MIT 协议进行发布。项目所随的部分源文件可能具备不同但与之兼容的许可协议。 pineapple-pictures-0.5.3/app/000077500000000000000000000000001415740431200161115ustar00rootroot00000000000000pineapple-pictures-0.5.3/app/aboutdialog.cpp000066400000000000000000000172461415740431200211210ustar00rootroot00000000000000#include "aboutdialog.h" #include #include #include #include #include #include #include #include #include AboutDialog::AboutDialog(QWidget *parent) : QDialog(parent) , m_tabWidget(new QTabWidget) , m_buttonBox(new QDialogButtonBox) , m_helpTextEdit(new QTextBrowser) , m_aboutTextEdit(new QTextBrowser) , m_specialThanksTextEdit(new QTextBrowser) , m_licenseTextEdit(new QTextBrowser) , m_3rdPartyLibsTextEdit(new QTextBrowser) { this->setWindowTitle(tr("About")); const QStringList helpStr { QStringLiteral("

%1

").arg(tr("Launch application with image file path as argument to load the file.")), QStringLiteral("

%1

").arg(tr("Drag and drop image file onto the window is also supported.")), QStringLiteral("

%1

").arg(tr("None of the operations in this application will alter the pictures on disk.")), QStringLiteral("

%1

").arg(tr("Context menu option explanation:")), QStringLiteral("
    "), // blumia: Chain two arg() here since it seems lupdate will remove one of them if we use // the old `arg(QCoreApp::translate(), tr())` way, but it's worth to mention // `arg(QCoreApp::translate(), this->tr())` works, but lupdate will complain about the usage. QStringLiteral("
  • %1:
    %2
  • ") .arg(QCoreApplication::translate("MainWindow", "Stay on top")) .arg(tr("Make window stay on top of all other windows.")), QStringLiteral("
  • %1:
    %2
  • ") .arg(QCoreApplication::translate("MainWindow", "Protected mode")) .arg(tr("Avoid close window accidentally. (eg. by double clicking the window)")), QStringLiteral("
") }; const QStringList aboutStr { QStringLiteral("

"), qApp->applicationDisplayName(), #ifdef GIT_DESCRIBE_VERSION_STRING (QStringLiteral("
") + tr("Version: %1").arg(GIT_DESCRIBE_VERSION_STRING)), #endif // GIT_DESCRIBE_VERSION_STRING QStringLiteral("
"), tr("Copyright (c) 2020 %1").arg(QStringLiteral("@BLumia")), QStringLiteral("
"), tr("Logo designed by %1").arg(QStringLiteral("@Lovelyblack")), QStringLiteral("
"), tr("Built with Qt %1 (%2)").arg(QT_VERSION_STR, QSysInfo::buildCpuArchitecture()), QStringLiteral("
%2").arg("https://github.com/BLumia/pineapple-pictures", tr("Source code")), QStringLiteral("
") }; QFile translaterHtml(":/plain/translators.html"); bool canOpenFile = translaterHtml.open(QIODevice::ReadOnly); const QByteArray & translatorList = canOpenFile ? translaterHtml.readAll() : ""; const QStringList specialThanksStr { QStringLiteral("

%1

%3

%4

").arg( tr("Contributors"), QStringLiteral("https://github.com/BLumia/pineapple-pictures/graphs/contributors"), tr("List of contributors on GitHub"), tr("Thanks to all people who contributed to this project.") ), QStringLiteral("

%1

%2

%3").arg( tr("Translators"), tr("I would like to thank the following people who volunteered to translate this application."), translatorList ) }; const QStringList licenseStr { QStringLiteral("

%1

").arg(tr("Your Rights")), QStringLiteral("

%1

%2

  • %3
  • %4
  • %5
  • %6
").arg( tr("%1 is released under the MIT License."), // %1 tr("This license grants people a number of freedoms:"), // %2 tr("You are free to use %1, for any purpose"), // %3 tr("You are free to distribute %1"), // %4 tr("You can study how %1 works and change it"), // %5 tr("You can distribute changed versions of %1") // %6 ).arg(QStringLiteral("%1")), QStringLiteral("

%1

").arg(tr("The MIT license guarantees you this freedom. Nobody is ever permitted to take it away.")), QStringLiteral("
%2
") }; const QString mitLicense(QStringLiteral(R"(Expat/MIT License Copyright (c) 2020 BLumia 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. )")); const QStringList thirdPartyLibsStr { QStringLiteral("

%1

").arg(tr("Third-party Libraries used by %1")), tr("%1 is built on the following free software libraries:", "Free as in freedom"), QStringLiteral("
    "), #ifdef HAVE_EXIV2_VERSION QStringLiteral("
  • %2: %3
  • ").arg("https://www.exiv2.org/", "Exiv2", "GPLv2"), #endif // EXIV2_VERSION QStringLiteral("
  • %2: %3
  • ").arg("https://www.qt.io/", "Qt", "GPLv2 + GPLv3 + LGPLv2.1 + LGPLv3"), QStringLiteral("
") }; m_helpTextEdit->setText(helpStr.join('\n')); m_aboutTextEdit->setText(aboutStr.join('\n')); m_aboutTextEdit->setOpenExternalLinks(true); m_specialThanksTextEdit->setText(specialThanksStr.join('\n')); m_specialThanksTextEdit->setOpenExternalLinks(true); m_licenseTextEdit->setText(licenseStr.join('\n').arg(qApp->applicationDisplayName(), mitLicense)); m_3rdPartyLibsTextEdit->setText(thirdPartyLibsStr.join('\n').arg(QStringLiteral("%1").arg(qApp->applicationDisplayName()))); m_3rdPartyLibsTextEdit->setOpenExternalLinks(true); m_tabWidget->addTab(m_helpTextEdit, tr("&Help")); m_tabWidget->addTab(m_aboutTextEdit, tr("&About")); m_tabWidget->addTab(m_specialThanksTextEdit, tr("&Special Thanks")); m_tabWidget->addTab(m_licenseTextEdit, tr("&License")); m_tabWidget->addTab(m_3rdPartyLibsTextEdit, tr("&Third-party Libraries")); m_buttonBox->setStandardButtons(QDialogButtonBox::Close); connect(m_buttonBox, QOverload::of(&QDialogButtonBox::clicked), this, [this](){ this->close(); }); setLayout(new QVBoxLayout); layout()->addWidget(m_tabWidget); layout()->addWidget(m_buttonBox); setMinimumSize(361, 161); // not sure why it complain "Unable to set geometry" setWindowFlag(Qt::WindowContextHelpButtonHint, false); } AboutDialog::~AboutDialog() { } QSize AboutDialog::sizeHint() const { return QSize(520, 350); } pineapple-pictures-0.5.3/app/aboutdialog.h000066400000000000000000000013511415740431200205540ustar00rootroot00000000000000#ifndef ABOUTDIALOG_H #define ABOUTDIALOG_H #include QT_BEGIN_NAMESPACE class QTextBrowser; class QTabWidget; class QDialogButtonBox; QT_END_NAMESPACE class AboutDialog : public QDialog { Q_OBJECT public: explicit AboutDialog(QWidget *parent = nullptr); ~AboutDialog() override; QSize sizeHint() const override; private: QTabWidget * m_tabWidget = nullptr; QDialogButtonBox * m_buttonBox = nullptr; QTextBrowser * m_helpTextEdit = nullptr; QTextBrowser * m_aboutTextEdit = nullptr; QTextBrowser * m_specialThanksTextEdit = nullptr; QTextBrowser * m_licenseTextEdit = nullptr; QTextBrowser * m_3rdPartyLibsTextEdit = nullptr; }; #endif // ABOUTDIALOG_H pineapple-pictures-0.5.3/app/actionmanager.cpp000066400000000000000000000123241415740431200214270ustar00rootroot00000000000000#include "actionmanager.h" #include "mainwindow.h" #include #define ICON_NAME(name)\ QStringLiteral(":/icons/" #name "") #define SETUP_NEW_ACTION(window, action)\ action->setObjectName(QString::fromUtf8( #action ));\ window->addAction(action); #define CREATE_NEW_ACTION(window, action)\ action = new QAction(window);\ SETUP_NEW_ACTION(window, action) #define CREATE_NEW_ICON_ACTION(window, action, iconname)\ action = new QAction(QIcon(ICON_NAME(iconname)), QString(), window);\ SETUP_NEW_ACTION(window, action) ActionManager::ActionManager() { } ActionManager::~ActionManager() { } void ActionManager::setupAction(MainWindow *mainWindow) { CREATE_NEW_ICON_ACTION(mainWindow, actionActualSize, zoom-original); CREATE_NEW_ICON_ACTION(mainWindow, actionToggleMaximize, view-fullscreen); CREATE_NEW_ICON_ACTION(mainWindow, actionZoomIn, zoom-in); CREATE_NEW_ICON_ACTION(mainWindow, actionZoomOut, zoom-out); CREATE_NEW_ICON_ACTION(mainWindow, actionToggleCheckerboard, view-background-checkerboard); CREATE_NEW_ICON_ACTION(mainWindow, actionRotateClockwise, object-rotate-right); CREATE_NEW_ACTION(mainWindow, actionPrevPicture); CREATE_NEW_ACTION(mainWindow, actionNextPicture); CREATE_NEW_ACTION(mainWindow, actionOpen); CREATE_NEW_ACTION(mainWindow, actionHorizontalFlip); CREATE_NEW_ACTION(mainWindow, actionFitInView); CREATE_NEW_ACTION(mainWindow, actionFitByWidth); CREATE_NEW_ACTION(mainWindow, actionCopyPixmap); CREATE_NEW_ACTION(mainWindow, actionCopyFilePath); CREATE_NEW_ACTION(mainWindow, actionPaste); CREATE_NEW_ACTION(mainWindow, actionToggleStayOnTop); CREATE_NEW_ACTION(mainWindow, actionToggleProtectMode); CREATE_NEW_ACTION(mainWindow, actionSettings); CREATE_NEW_ACTION(mainWindow, actionHelp); CREATE_NEW_ACTION(mainWindow, actionProperties); CREATE_NEW_ACTION(mainWindow, actionQuitApp); retranslateUi(mainWindow); QMetaObject::connectSlotsByName(mainWindow); } void ActionManager::retranslateUi(MainWindow *mainWindow) { Q_UNUSED(mainWindow); actionOpen->setText(QCoreApplication::translate("MainWindow", "&Open...", nullptr)); actionActualSize->setText(QCoreApplication::translate("MainWindow", "Actual size", nullptr)); actionToggleMaximize->setText(QCoreApplication::translate("MainWindow", "Toggle maximize", nullptr)); actionZoomIn->setText(QCoreApplication::translate("MainWindow", "Zoom in", nullptr)); actionZoomOut->setText(QCoreApplication::translate("MainWindow", "Zoom out", nullptr)); actionToggleCheckerboard->setText(QCoreApplication::translate("MainWindow", "Toggle Checkerboard", nullptr)); actionRotateClockwise->setText(QCoreApplication::translate("MainWindow", "Rotate right", nullptr)); actionPrevPicture->setText(QCoreApplication::translate("MainWindow", "Previous image", nullptr)); actionNextPicture->setText(QCoreApplication::translate("MainWindow", "Next image", nullptr)); actionHorizontalFlip->setText(QCoreApplication::translate("MainWindow", "Flip &Horizontally", nullptr)); actionFitInView->setText("Fit in view"); // TODO: what should it called? actionFitByWidth->setText("Fit by width"); // TODO: what should it called? actionCopyPixmap->setText(QCoreApplication::translate("MainWindow", "Copy P&ixmap", nullptr)); actionCopyFilePath->setText(QCoreApplication::translate("MainWindow", "Copy &File Path", nullptr)); actionPaste->setText(QCoreApplication::translate("MainWindow", "&Paste", nullptr)); actionToggleStayOnTop->setText(QCoreApplication::translate("MainWindow", "Stay on top", nullptr)); actionToggleProtectMode->setText(QCoreApplication::translate("MainWindow", "Protected mode", nullptr)); actionSettings->setText(QCoreApplication::translate("MainWindow", "Configure...", nullptr)); actionHelp->setText(QCoreApplication::translate("MainWindow", "Help", nullptr)); actionProperties->setText(QCoreApplication::translate("MainWindow", "Properties", nullptr)); actionQuitApp->setText(QCoreApplication::translate("MainWindow", "Quit", nullptr)); } void ActionManager::setupShortcuts() { actionOpen->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_O)); actionActualSize->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_0)); actionZoomIn->setShortcut(QKeySequence(QKeySequence::ZoomIn)); actionZoomOut->setShortcut(QKeySequence(QKeySequence::ZoomOut)); actionPrevPicture->setShortcuts({ QKeySequence(Qt::Key_PageUp), QKeySequence(Qt::Key_Left), }); actionNextPicture->setShortcuts({ QKeySequence(Qt::Key_PageDown), QKeySequence(Qt::Key_Right), }); actionHorizontalFlip->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_R)); actionCopyPixmap->setShortcut(QKeySequence(QKeySequence::Copy)); actionPaste->setShortcut(QKeySequence::Paste); actionHelp->setShortcut(QKeySequence::HelpContents); actionSettings->setShortcut(QKeySequence::Preferences); actionProperties->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_I)); actionQuitApp->setShortcuts({ QKeySequence(Qt::Key_Space), QKeySequence(Qt::Key_Escape) }); } pineapple-pictures-0.5.3/app/actionmanager.h000066400000000000000000000020051415740431200210670ustar00rootroot00000000000000#ifndef ACTIONMANAGER_H #define ACTIONMANAGER_H #include class MainWindow; class ActionManager { public: ActionManager(); ~ActionManager(); void setupAction(MainWindow * mainWindow); void retranslateUi(MainWindow *MainWindow); void setupShortcuts(); public: QAction *actionOpen; QAction *actionActualSize; QAction *actionToggleMaximize; QAction *actionZoomIn; QAction *actionZoomOut; QAction *actionToggleCheckerboard; QAction *actionRotateClockwise; QAction *actionPrevPicture; QAction *actionNextPicture; QAction *actionHorizontalFlip; QAction *actionFitInView; QAction *actionFitByWidth; QAction *actionCopyPixmap; QAction *actionCopyFilePath; QAction *actionPaste; QAction *actionToggleStayOnTop; QAction *actionToggleProtectMode; QAction *actionSettings; QAction *actionHelp; QAction *actionProperties; QAction *actionQuitApp; }; #endif // ACTIONMANAGER_H pineapple-pictures-0.5.3/app/bottombuttongroup.cpp000066400000000000000000000033451415740431200224370ustar00rootroot00000000000000#include "bottombuttongroup.h" #include "opacityhelper.h" #include #include #include #include BottomButtonGroup::BottomButtonGroup(const std::vector &actionList, QWidget *parent) : QGroupBox (parent) , m_opacityHelper(new OpacityHelper(this)) { QHBoxLayout * mainLayout = new QHBoxLayout(this); mainLayout->setSizeConstraint(QLayout::SetFixedSize); this->setLayout(mainLayout); this->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Minimum); this->setStyleSheet("BottomButtonGroup {" "border: 1px solid gray;" "border-top-left-radius: 10px;" "border-top-right-radius: 10px;" "border-style: none;" "background-color:rgba(0,0,0,120)" "}" "QToolButton {" "background:transparent;" "}" "QToolButton:!focus {" "border-style: none;" "}"); auto newActionBtn = [this](QAction * action) -> QToolButton * { QToolButton * btn = new QToolButton(this); btn->setDefaultAction(action); btn->setIconSize(QSize(40, 40)); btn->setFixedSize(40, 40); return btn; }; for (QAction * action : actionList) { addButton(newActionBtn(action)); } } void BottomButtonGroup::setOpacity(qreal opacity, bool animated) { m_opacityHelper->setOpacity(opacity, animated); } void BottomButtonGroup::addButton(QAbstractButton *button) { layout()->addWidget(button); updateGeometry(); } pineapple-pictures-0.5.3/app/bottombuttongroup.h000066400000000000000000000010121415740431200220710ustar00rootroot00000000000000#ifndef BOTTOMBUTTONGROUP_H #define BOTTOMBUTTONGROUP_H #include #include #include class OpacityHelper; class BottomButtonGroup : public QGroupBox { Q_OBJECT public: explicit BottomButtonGroup(const std::vector & actionList, QWidget *parent = nullptr); void setOpacity(qreal opacity, bool animated = true); void addButton(QAbstractButton *button); private: OpacityHelper * m_opacityHelper; }; #endif // BOTTOMBUTTONGROUP_H pineapple-pictures-0.5.3/app/exiv2wrapper.cpp000066400000000000000000000062161415740431200212600ustar00rootroot00000000000000#include "exiv2wrapper.h" #ifdef HAVE_EXIV2_VERSION #include #else // HAVE_EXIV2_VERSION namespace Exiv2 { class Image {}; } #endif // HAVE_EXIV2_VERSION #include #include #include Exiv2Wrapper::Exiv2Wrapper() { } Exiv2Wrapper::~Exiv2Wrapper() { } template void Exiv2Wrapper::cacheSection(Collection collection) { const Collection& exifData = collection; Iterator it = exifData.begin(), end = exifData.end(); for (; it != end; ++it) { QString key = QString::fromUtf8(it->key().c_str()); if (it->tagName().substr(0, 2) == "0x") continue; QString label = QString::fromLocal8Bit(it->tagLabel().c_str()); std::ostringstream stream; stream << *it; QString value = QString::fromUtf8(stream.str().c_str()); m_metadataValue.insert(key, value); m_metadataLabel.insert(key, label); qDebug() << key << label << value; } } bool Exiv2Wrapper::load(const QString &filePath) { #ifdef HAVE_EXIV2_VERSION QByteArray filePathByteArray = QFile::encodeName(filePath); try { m_exivImage.reset(Exiv2::ImageFactory::open(filePathByteArray.constData()).release()); m_exivImage->readMetadata(); } catch (const Exiv2::Error& error) { m_errMsg = QString::fromUtf8(error.what()); return false; } return true; #else // HAVE_EXIV2_VERSION Q_UNUSED(filePath); return false; #endif // HAVE_EXIV2_VERSION } void Exiv2Wrapper::cacheSections() { #ifdef HAVE_EXIV2_VERSION if (m_exivImage->checkMode(Exiv2::mdExif) & Exiv2::amRead) { cacheSection(m_exivImage->exifData()); } if (m_exivImage->checkMode(Exiv2::mdIptc) & Exiv2::amRead) { cacheSection(m_exivImage->iptcData()); } if (m_exivImage->checkMode(Exiv2::mdXmp) & Exiv2::amRead) { cacheSection(m_exivImage->xmpData()); } // qDebug() << m_metadataValue; // qDebug() << m_metadataLabel; #endif // HAVE_EXIV2_VERSION } QString Exiv2Wrapper::comment() const { #ifdef HAVE_EXIV2_VERSION return m_exivImage->comment().c_str(); #else // HAVE_EXIV2_VERSION return QString(); #endif // HAVE_EXIV2_VERSION } QString Exiv2Wrapper::label(const QString &key) const { return m_metadataLabel.value(key); } QString Exiv2Wrapper::value(const QString &key) const { return m_metadataValue.value(key); } QString Exiv2Wrapper::XmpValue(const QString &rawValue) { QString ignored; return Exiv2Wrapper::XmpValue(rawValue, ignored); } QString Exiv2Wrapper::XmpValue(const QString &rawValue, QString &language) { if (rawValue.size() > 6 && rawValue.startsWith(QLatin1String("lang=\""))) { int pos = rawValue.indexOf('"', 6); if (pos != -1) { language = rawValue.mid(6, pos - 6); return (rawValue.mid(pos + 2)); } } language.clear(); return rawValue; } pineapple-pictures-0.5.3/app/exiv2wrapper.h000066400000000000000000000015331415740431200207220ustar00rootroot00000000000000#ifndef EXIV2WRAPPER_H #define EXIV2WRAPPER_H #include #include #include namespace Exiv2 { class Image; } class Exiv2Wrapper { public: Exiv2Wrapper(); ~Exiv2Wrapper(); bool load(const QString& filePath); void cacheSections(); QString comment() const; QString label(const QString & key) const; QString value(const QString & key) const; static QString XmpValue(const QString &rawValue); static QString XmpValue(const QString &rawValue, QString & language); private: std::unique_ptr m_exivImage; QMap m_metadataValue; QMap m_metadataLabel; QString m_errMsg; template void cacheSection(Collection collection); }; #endif // EXIV2WRAPPER_H pineapple-pictures-0.5.3/app/framelesswindow.cpp000066400000000000000000000076511415740431200220370ustar00rootroot00000000000000#include "framelesswindow.h" #include #include #ifdef _WIN32 #include #endif // _WIN32 FramelessWindow::FramelessWindow(QWidget *parent) : QWidget(parent) , m_centralLayout(new QVBoxLayout(this)) { // We should use Qt::WindowMinMaxButtonsHint here but there is a bug in Qt // that will make pressing Meta+Up cause the app fullscreen under Windows, // so for now we only use the Qt::WindowMinimizeButtonHint flag here. // https://bugreports.qt.io/browse/QTBUG-91226 this->setWindowFlags(Qt::Window | Qt::FramelessWindowHint | Qt::WindowMinimizeButtonHint); m_centralLayout->setContentsMargins(QMargins()); } void FramelessWindow::setCentralWidget(QWidget *widget) { if (m_centralWidget) { m_centralLayout->removeWidget(m_centralWidget); m_centralWidget->deleteLater(); } m_centralLayout->addWidget(widget); m_centralWidget = widget; } bool FramelessWindow::nativeEvent(const QByteArray &eventType, void *message, NATIVE_RESULT *result) { #ifdef _WIN32 // https://stackoverflow.com/questions/43505580/qt-windows-resizable-frameless-window // Too lazy to do this now.. just stackoverflow it and did a copy and paste.. Q_UNUSED(eventType) MSG* msg = static_cast(message); if (msg->message == WM_NCHITTEST) { if (isMaximized()) { return false; } *result = 0; const LONG borderWidth = 8; RECT winrect; GetWindowRect(reinterpret_cast(winId()), &winrect); // must be short to correctly work with multiple monitors (negative coordinates) short x = msg->lParam & 0x0000FFFF; short y = (msg->lParam & 0xFFFF0000) >> 16; bool resizeWidth = minimumWidth() != maximumWidth(); bool resizeHeight = minimumHeight() != maximumHeight(); if (resizeWidth) { //left border if (x >= winrect.left && x < winrect.left + borderWidth) { *result = HTLEFT; } //right border if (x < winrect.right && x >= winrect.right - borderWidth) { *result = HTRIGHT; } } if (resizeHeight) { //bottom border if (y < winrect.bottom && y >= winrect.bottom - borderWidth) { *result = HTBOTTOM; } //top border if (y >= winrect.top && y < winrect.top + borderWidth) { *result = HTTOP; } } if (resizeWidth && resizeHeight) { //bottom left corner if (x >= winrect.left && x < winrect.left + borderWidth && y < winrect.bottom && y >= winrect.bottom - borderWidth) { *result = HTBOTTOMLEFT; } //bottom right corner if (x < winrect.right && x >= winrect.right - borderWidth && y < winrect.bottom && y >= winrect.bottom - borderWidth) { *result = HTBOTTOMRIGHT; } //top left corner if (x >= winrect.left && x < winrect.left + borderWidth && y >= winrect.top && y < winrect.top + borderWidth) { *result = HTTOPLEFT; } //top right corner if (x < winrect.right && x >= winrect.right - borderWidth && y >= winrect.top && y < winrect.top + borderWidth) { *result = HTTOPRIGHT; } } if (*result != 0) return true; QWidget *action = QApplication::widgetAt(QCursor::pos()); if (action == this) { *result = HTCAPTION; return true; } } return false; #else return QWidget::nativeEvent(eventType, message, result); #endif // _WIN32 } pineapple-pictures-0.5.3/app/framelesswindow.h000066400000000000000000000013751415740431200215010ustar00rootroot00000000000000#ifndef FRAMELESSWINDOW_H #define FRAMELESSWINDOW_H #include #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) typedef qintptr NATIVE_RESULT; #else typedef long NATIVE_RESULT; #endif // QT_VERSION_CHECK(6, 0, 0) QT_BEGIN_NAMESPACE class QVBoxLayout; QT_END_NAMESPACE class FramelessWindow : public QWidget { Q_OBJECT public: explicit FramelessWindow(QWidget *parent = nullptr); void setCentralWidget(QWidget * widget); protected: bool nativeEvent(const QByteArray& eventType, void* message, NATIVE_RESULT* result) override; private: QVBoxLayout * m_centralLayout = nullptr; QWidget * m_centralWidget = nullptr; // just a pointer, doesn't take the ownership. }; #endif // FRAMELESSWINDOW_H pineapple-pictures-0.5.3/app/graphicsscene.cpp000066400000000000000000000067211415740431200214410ustar00rootroot00000000000000#include "graphicsscene.h" #include #include #include #include #include #include #include #include #include class PGraphicsPixmapItem : public QGraphicsPixmapItem { public: PGraphicsPixmapItem(const QPixmap &pixmap, QGraphicsItem *parent = nullptr) : QGraphicsPixmapItem(pixmap, parent) {} void setScaleHint(float scaleHint) { m_scaleHint = scaleHint; } void paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget) override { if (transformationMode() == Qt::FastTransformation) { return QGraphicsPixmapItem::paint(painter, option, widget); } else { // painter->setRenderHints(QPainter::Antialiasing); QSizeF resizedScale(boundingRect().size()); resizedScale *= m_scaleHint; painter->drawPixmap(QRectF(offset(), boundingRect().size()).toRect(), pixmap().scaled(resizedScale.toSize(), Qt::KeepAspectRatio, Qt::SmoothTransformation) ); } } private: float m_scaleHint = 1; }; GraphicsScene::GraphicsScene(QObject *parent) : QGraphicsScene(parent) { showText(tr("Drag image here")); } GraphicsScene::~GraphicsScene() { } void GraphicsScene::showImage(const QPixmap &pixmap) { this->clear(); PGraphicsPixmapItem * pixmapItem = new PGraphicsPixmapItem(pixmap); this->addItem(pixmapItem); pixmapItem->setShapeMode(QGraphicsPixmapItem::BoundingRectShape); m_theThing = pixmapItem; this->setSceneRect(m_theThing->boundingRect()); } void GraphicsScene::showText(const QString &text) { this->clear(); QGraphicsTextItem * textItem = this->addText(text); textItem->setDefaultTextColor(QColor("White")); m_theThing = textItem; this->setSceneRect(m_theThing->boundingRect()); } void GraphicsScene::showSvg(const QString &filepath) { this->clear(); QGraphicsSvgItem * svgItem = new QGraphicsSvgItem(filepath); this->addItem(svgItem); m_theThing = svgItem; this->setSceneRect(m_theThing->boundingRect()); } void GraphicsScene::showAnimated(const QString &filepath) { this->clear(); QLabel * label = new QLabel; QMovie * movie = new QMovie(filepath, QByteArray(), label); label->setStyleSheet("background-color:rgba(225,255,255,0);"); label->setMovie(movie); this->addWidget(label); movie->start(); m_theThing = this->addRect(QRect(QPoint(0, 0), label->sizeHint()), QPen(Qt::transparent)); this->setSceneRect(m_theThing->boundingRect()); } bool GraphicsScene::trySetTransformationMode(Qt::TransformationMode mode, float scaleHint) { PGraphicsPixmapItem * pixmapItem = qgraphicsitem_cast(m_theThing); if (pixmapItem) { pixmapItem->setTransformationMode(mode); pixmapItem->setScaleHint(scaleHint); return true; } return false; } QPixmap GraphicsScene::renderToPixmap() { PGraphicsPixmapItem * pixmapItem = qgraphicsitem_cast(m_theThing); if (pixmapItem) { return pixmapItem->pixmap(); } QPixmap pixmap(sceneRect().toRect().size()); pixmap.fill(Qt::transparent); QPainter p(&pixmap); render(&p, sceneRect()); return pixmap; } pineapple-pictures-0.5.3/app/graphicsscene.h000066400000000000000000000010731415740431200211010ustar00rootroot00000000000000#ifndef GRAPHICSSCENE_H #define GRAPHICSSCENE_H #include class GraphicsScene : public QGraphicsScene { Q_OBJECT public: GraphicsScene(QObject *parent = nullptr); ~GraphicsScene(); void showImage(const QPixmap &pixmap); void showText(const QString &text); void showSvg(const QString &filepath); void showAnimated(const QString &filepath); bool trySetTransformationMode(Qt::TransformationMode mode, float scaleHint); QPixmap renderToPixmap(); private: QGraphicsItem * m_theThing; }; #endif // GRAPHICSSCENE_H pineapple-pictures-0.5.3/app/graphicsview.cpp000066400000000000000000000305361415740431200213170ustar00rootroot00000000000000#include "graphicsview.h" #include "graphicsscene.h" #include #include #include #include #include #include // TODO: remove this once we drop older Qt support. #if QT_VERSION >= QT_VERSION_CHECK(5, 13, 0) #define COMPAT_CONSTCOLOR constexpr #else #define COMPAT_CONSTCOLOR const #endif GraphicsView::GraphicsView(QWidget *parent) : QGraphicsView (parent) { setDragMode(QGraphicsView::ScrollHandDrag); setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setResizeAnchor(QGraphicsView::AnchorUnderMouse); setTransformationAnchor(QGraphicsView::AnchorUnderMouse); setStyleSheet("background-color: rgba(0, 0, 0, 220);" "border-radius: 3px;"); setAcceptDrops(true); setCheckerboardEnabled(false); connect(horizontalScrollBar(), &QScrollBar::valueChanged, this, &GraphicsView::viewportRectChanged); connect(verticalScrollBar(), &QScrollBar::valueChanged, this, &GraphicsView::viewportRectChanged); } void GraphicsView::showFileFromPath(const QString &filePath, bool doRequestGallery) { emit navigatorViewRequired(false, transform()); if (filePath.endsWith(".svg")) { showSvg(filePath); } else { QImageReader imageReader(filePath); imageReader.setAutoTransform(true); imageReader.setDecideFormatFromContent(true); // Since if the image format / plugin does not support this feature, imageFormat() will returns an invalid format. // So we cannot use imageFormat() and check if it returns QImage::Format_Invalid to detect if we support the file. // QImage::Format imageFormat = imageReader.imageFormat(); if (imageReader.format().isEmpty()) { doRequestGallery = false; showText(tr("File is not a valid image")); } else if (imageReader.supportsAnimation() && imageReader.imageCount() > 1) { showAnimated(filePath); } else if (!imageReader.canRead()) { doRequestGallery = false; showText(tr("Image data is invalid or currently unsupported")); } else { const QPixmap & pixmap = QPixmap::fromImageReader(&imageReader); if (pixmap.isNull()) { doRequestGallery = false; showText(tr("Image data is invalid or currently unsupported")); } else { showImage(pixmap); } } } if (doRequestGallery) { emit requestGallery(filePath); } } void GraphicsView::showImage(const QPixmap &pixmap) { resetTransform(); scene()->showImage(pixmap); displayScene(); } void GraphicsView::showImage(const QImage &image) { resetTransform(); scene()->showImage(QPixmap::fromImage(image)); displayScene(); } void GraphicsView::showText(const QString &text) { resetTransform(); scene()->showText(text); displayScene(); } void GraphicsView::showSvg(const QString &filepath) { resetTransform(); scene()->showSvg(filepath); displayScene(); } void GraphicsView::showAnimated(const QString &filepath) { resetTransform(); scene()->showAnimated(filepath); displayScene(); } GraphicsScene *GraphicsView::scene() const { return qobject_cast(QGraphicsView::scene()); } void GraphicsView::setScene(GraphicsScene *scene) { return QGraphicsView::setScene(scene); } qreal GraphicsView::scaleFactor() const { return QStyleOptionGraphicsItem::levelOfDetailFromTransform(transform()); } void GraphicsView::resetTransform() { QGraphicsView::resetTransform(); } void GraphicsView::zoomView(qreal scaleFactor) { m_enableFitInView = false; scale(scaleFactor, scaleFactor); applyTransformationModeByScaleFactor(); emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform()); } // This is always according to user's view. // the direction of the rotation will NOT be clockwise because the y-axis points downwards. void GraphicsView::rotateView(bool clockwise) { resetScale(); QTransform tf(0, clockwise ? 1 : -1, 0, clockwise ? -1 : 1, 0, 0, 0, 0, 1); tf = transform() * tf; setTransform(tf); } void GraphicsView::flipView(bool horizontal) { QTransform tf(horizontal ? -1 : 1, 0, 0, 0, horizontal ? 1 : -1, 0, 0, 0, 1); tf = transform() * tf; setTransform(tf); // Ensure the navigation view is also flipped. emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform()); } void GraphicsView::resetScale() { setTransform(resetScale(transform())); applyTransformationModeByScaleFactor(); emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform()); } void GraphicsView::fitInView(const QRectF &rect, Qt::AspectRatioMode aspectRadioMode) { QGraphicsView::fitInView(rect, aspectRadioMode); applyTransformationModeByScaleFactor(); } void GraphicsView::fitByOrientation(Qt::Orientation ori, bool scaleDownOnly) { resetScale(); QRectF viewRect = this->viewport()->rect().adjusted(2, 2, -2, -2); QRectF imageRect = transform().mapRect(sceneRect()); qreal ratio; if (ori == Qt::Horizontal) { ratio = viewRect.width() / imageRect.width(); } else { ratio = viewRect.height() / imageRect.height(); } if (scaleDownOnly && ratio > 1) ratio = 1; scale(ratio, ratio); centerOn(imageRect.top(), 0); m_enableFitInView = false; applyTransformationModeByScaleFactor(); emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform()); } void GraphicsView::displayScene() { if (isSceneBiggerThanView()) { fitInView(sceneRect(), Qt::KeepAspectRatio); } m_enableFitInView = true; } bool GraphicsView::isSceneBiggerThanView() const { if (!isThingSmallerThanWindowWith(transform())) { return true; } else { return false; } } // Automately do fit in view when viewport(window) smaller than image original size. void GraphicsView::setEnableAutoFitInView(bool enable) { m_enableFitInView = enable; } inline double zeroOrOne(double number) { return qFuzzyIsNull(number) ? 0 : (number > 0 ? 1 : -1); } // Note: this only works if we only have 90 degree based rotation // and no shear/translate. QTransform GraphicsView::resetScale(const QTransform & orig) { return QTransform(zeroOrOne(orig.m11()), zeroOrOne(orig.m12()), zeroOrOne(orig.m21()), zeroOrOne(orig.m22()), orig.dx(), orig.dy()); } void GraphicsView::toggleCheckerboard(bool invertCheckerboardColor) { setCheckerboardEnabled(!m_checkerboardEnabled, invertCheckerboardColor); } void GraphicsView::mousePressEvent(QMouseEvent *event) { if (shouldIgnoreMousePressMoveEvent(event)) { event->ignore(); // blumia: return here, or the QMouseEvent event transparency won't // work if we set a QGraphicsView::ScrollHandDrag drag mode. return; } return QGraphicsView::mousePressEvent(event); } void GraphicsView::mouseMoveEvent(QMouseEvent *event) { if (shouldIgnoreMousePressMoveEvent(event)) { event->ignore(); } return QGraphicsView::mouseMoveEvent(event); } void GraphicsView::mouseReleaseEvent(QMouseEvent *event) { if (event->button() == Qt::ForwardButton || event->button() == Qt::BackButton) { event->ignore(); } else { QGraphicsItem *item = itemAt(event->pos()); if (!item) { event->ignore(); } } return QGraphicsView::mouseReleaseEvent(event); } void GraphicsView::wheelEvent(QWheelEvent *event) { event->ignore(); // blumia: no need for calling parent method. } void GraphicsView::resizeEvent(QResizeEvent *event) { if (m_enableFitInView) { bool originalSizeSmallerThanWindow = isThingSmallerThanWindowWith(resetScale(transform())); if (originalSizeSmallerThanWindow && scaleFactor() >= 1) { // no longer need to do fitInView() // but we leave the m_enableFitInView value unchanged in case // user resize down the window again. } else if (originalSizeSmallerThanWindow && scaleFactor() < 1) { resetScale(); } else { fitInView(sceneRect(), Qt::KeepAspectRatio); } } else { emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform()); } return QGraphicsView::resizeEvent(event); } void GraphicsView::dragEnterEvent(QDragEnterEvent *event) { if (event->mimeData()->hasUrls() || event->mimeData()->hasImage() || event->mimeData()->hasText()) { event->acceptProposedAction(); } else { event->ignore(); } // qDebug() << event->mimeData() << "Drag Enter Event" // << event->mimeData()->hasUrls() << event->mimeData()->hasImage() // << event->mimeData()->formats() << event->mimeData()->hasFormat("text/uri-list"); return QGraphicsView::dragEnterEvent(event); } void GraphicsView::dragMoveEvent(QDragMoveEvent *event) { Q_UNUSED(event) // by default, QGraphicsView/Scene will ignore the action if there are no QGraphicsItem under cursor. // We actually doesn't care and would like to keep the drag event as-is, so just do nothing here. } void GraphicsView::dropEvent(QDropEvent *event) { event->acceptProposedAction(); const QMimeData * mimeData = event->mimeData(); if (mimeData->hasUrls()) { const QList &urls = mimeData->urls(); if (urls.isEmpty()) { showText(tr("File url list is empty")); } else { showFileFromPath(urls.first().toLocalFile(), true); } } else if (mimeData->hasImage()) { QImage img = qvariant_cast(mimeData->imageData()); QPixmap pixmap = QPixmap::fromImage(img); if (pixmap.isNull()) { showText(tr("Image data is invalid")); } else { showImage(pixmap); } } else if (mimeData->hasText()) { showText(mimeData->text()); } else { showText(tr("Not supported mimedata: %1").arg(mimeData->formats().first())); } } bool GraphicsView::isThingSmallerThanWindowWith(const QTransform &transform) const { return rect().size().expandedTo(transform.mapRect(sceneRect()).size().toSize()) == rect().size(); } bool GraphicsView::shouldIgnoreMousePressMoveEvent(const QMouseEvent *event) const { if (event->buttons() == Qt::NoButton) { return true; } QGraphicsItem *item = itemAt(event->pos()); if (!item) { return true; } if (isThingSmallerThanWindowWith(transform())) { return true; } return false; } void GraphicsView::setCheckerboardEnabled(bool enabled, bool invertColor) { m_checkerboardEnabled = enabled; m_isLastCheckerboardColorInverted = invertColor; if (m_checkerboardEnabled) { // Prepare background check-board pattern QPixmap tilePixmap(0x20, 0x20); tilePixmap.fill(invertColor ? QColor(220, 220, 220, 170) : QColor(35, 35, 35, 170)); QPainter tilePainter(&tilePixmap); COMPAT_CONSTCOLOR QColor color(45, 45, 45, 170); COMPAT_CONSTCOLOR QColor invertedColor(210, 210, 210, 170); tilePainter.fillRect(0, 0, 0x10, 0x10, invertColor ? invertedColor : color); tilePainter.fillRect(0x10, 0x10, 0x10, 0x10, invertColor ? invertedColor : color); tilePainter.end(); setBackgroundBrush(tilePixmap); } else { setBackgroundBrush(Qt::transparent); } } void GraphicsView::applyTransformationModeByScaleFactor() { if (this->scaleFactor() < 1) { scene()->trySetTransformationMode(Qt::SmoothTransformation, this->scaleFactor()); } else { scene()->trySetTransformationMode(Qt::FastTransformation, this->scaleFactor()); } } pineapple-pictures-0.5.3/app/graphicsview.h000066400000000000000000000050611415740431200207570ustar00rootroot00000000000000#ifndef GRAPHICSVIEW_H #define GRAPHICSVIEW_H #include #include class GraphicsScene; class GraphicsView : public QGraphicsView { Q_OBJECT public: GraphicsView(QWidget *parent = nullptr); void showFileFromPath(const QString &filePath, bool requestGallery = false); void showImage(const QPixmap &pixmap); void showImage(const QImage &image); void showText(const QString &text); void showSvg(const QString &filepath); void showAnimated(const QString &filepath); GraphicsScene * scene() const; void setScene(GraphicsScene *scene); qreal scaleFactor() const; void resetTransform(); void zoomView(qreal scaleFactor); void rotateView(bool clockwise = true); void flipView(bool horizontal = true); void resetScale(); void fitInView(const QRectF &rect, Qt::AspectRatioMode aspectRadioMode = Qt::IgnoreAspectRatio); void fitByOrientation(Qt::Orientation ori = Qt::Horizontal, bool scaleDownOnly = false); void displayScene(); bool isSceneBiggerThanView() const; void setEnableAutoFitInView(bool enable = true); static QTransform resetScale(const QTransform & orig); signals: void navigatorViewRequired(bool required, QTransform transform); void viewportRectChanged(); void requestGallery(const QString &filePath); public slots: void toggleCheckerboard(bool invertCheckerboardColor = false); private: void mousePressEvent(QMouseEvent * event) override; void mouseMoveEvent(QMouseEvent * event) override; void mouseReleaseEvent(QMouseEvent * event) override; void wheelEvent(QWheelEvent *event) override; void resizeEvent(QResizeEvent *event) override; void dragEnterEvent(QDragEnterEvent *event) override; void dragMoveEvent(QDragMoveEvent *event) override; void dropEvent(QDropEvent *event) override; bool isThingSmallerThanWindowWith(const QTransform &transform) const; bool shouldIgnoreMousePressMoveEvent(const QMouseEvent *event) const; void setCheckerboardEnabled(bool enabled, bool invertColor = false); void applyTransformationModeByScaleFactor(); // Consider switch to 3 state for "no fit", "always fit" and "fit when view is smaller"? // ... or even more? e.g. "fit/snap width" things... // Currently it's "no fit" when it's false and "fit when view is smaller" when it's true. bool m_enableFitInView = false; bool m_checkerboardEnabled = false; bool m_isLastCheckerboardColorInverted = false; }; #endif // GRAPHICSVIEW_H pineapple-pictures-0.5.3/app/main.cpp000066400000000000000000000026671415740431200175540ustar00rootroot00000000000000#include "mainwindow.h" #include "playlistmanager.h" #include #include #include #include #include // QM_FILE_INSTALL_DIR should be defined from the CMakeLists file. #ifndef QM_FILE_INSTALL_DIR #define QM_FILE_INSTALL_DIR ":/i18n/" #endif // QM_FILE_INSTALL_DIR int main(int argc, char *argv[]) { QApplication a(argc, argv); QTranslator translator; QString qmDir; #ifdef _WIN32 qmDir = QDir(QCoreApplication::applicationDirPath()).absoluteFilePath("translations"); #else qmDir = QT_STRINGIFY(QM_FILE_INSTALL_DIR); #endif if (translator.load(QString("PineapplePictures_%1").arg(QLocale::system().name()), qmDir)) { a.installTranslator(&translator); } a.setApplicationName("Pineapple Pictures"); a.setApplicationDisplayName(QCoreApplication::translate("main", "Pineapple Pictures")); // parse commandline arguments QCommandLineParser parser; parser.addPositionalArgument("File list", QCoreApplication::translate("main", "File list.")); parser.addHelpOption(); parser.process(a); MainWindow w; w.show(); QStringList urlStrList = parser.positionalArguments(); QList && urlList = PlaylistManager::convertToUrlList(urlStrList); if (!urlList.isEmpty()) { w.showUrls(urlList); w.adjustWindowSizeBySceneRect(); } return a.exec(); } pineapple-pictures-0.5.3/app/mainwindow.cpp000066400000000000000000000515141415740431200207770ustar00rootroot00000000000000#include "mainwindow.h" #include "settings.h" #include "toolbutton.h" #include "bottombuttongroup.h" #include "graphicsview.h" #include "navigatorview.h" #include "graphicsscene.h" #include "settingsdialog.h" #include "aboutdialog.h" #include "metadatamodel.h" #include "metadatadialog.h" #include "actionmanager.h" #include "playlistmanager.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include MainWindow::MainWindow(QWidget *parent) : FramelessWindow(parent) , m_am(new ActionManager) , m_pm(new PlaylistManager(PlaylistManager::PL_SAMEFOLDER, this)) { if (Settings::instance()->stayOnTop()) { this->setWindowFlag(Qt::WindowStaysOnTopHint); } this->setAttribute(Qt::WA_TranslucentBackground, true); this->setMinimumSize(350, 330); this->setWindowIcon(QIcon(":/icons/app-icon.svg")); this->setMouseTracking(true); m_pm->setAutoLoadFilterSuffix({"*.jpg", "*.jpeg", "*.jfif", "*.png", "*.gif", "*.svg", "*.bmp", "*.webp"}); m_fadeOutAnimation = new QPropertyAnimation(this, "windowOpacity"); m_fadeOutAnimation->setDuration(300); m_fadeOutAnimation->setStartValue(1); m_fadeOutAnimation->setEndValue(0); m_floatUpAnimation = new QPropertyAnimation(this, "geometry"); m_floatUpAnimation->setDuration(300); m_floatUpAnimation->setEasingCurve(QEasingCurve::OutCirc); m_exitAnimationGroup = new QParallelAnimationGroup(this); m_exitAnimationGroup->addAnimation(m_fadeOutAnimation); m_exitAnimationGroup->addAnimation(m_floatUpAnimation); connect(m_exitAnimationGroup, &QParallelAnimationGroup::finished, this, &QWidget::close); GraphicsScene * scene = new GraphicsScene(this); m_graphicsView = new GraphicsView(this); m_graphicsView->setScene(scene); this->setCentralWidget(m_graphicsView); m_gv = new NavigatorView(this); m_gv->setFixedSize(220, 160); m_gv->setScene(scene); m_gv->setMainView(m_graphicsView); m_gv->fitInView(m_gv->sceneRect(), Qt::KeepAspectRatio); connect(m_graphicsView, &GraphicsView::navigatorViewRequired, this, [ = ](bool required, QTransform tf){ m_gv->setTransform(GraphicsView::resetScale(tf)); m_gv->fitInView(m_gv->sceneRect(), Qt::KeepAspectRatio); m_gv->setVisible(required); m_gv->updateMainViewportRegion(); }); connect(m_graphicsView, &GraphicsView::viewportRectChanged, m_gv, &NavigatorView::updateMainViewportRegion); connect(m_graphicsView, &GraphicsView::requestGallery, this, &MainWindow::loadGalleryBySingleLocalFile); m_closeButton = new ToolButton(true, m_graphicsView); m_closeButton->setIcon(QIcon(":/icons/window-close")); m_closeButton->setIconSize(QSize(50, 50)); connect(m_closeButton, &QAbstractButton::clicked, this, &MainWindow::closeWindow); m_prevButton = new ToolButton(false, m_graphicsView); m_prevButton->setIcon(QIcon(":/icons/go-previous")); m_prevButton->setIconSize(QSize(75, 75)); m_prevButton->setVisible(false); m_prevButton->setOpacity(0, false); m_nextButton = new ToolButton(false, m_graphicsView); m_nextButton->setIcon(QIcon(":/icons/go-next")); m_nextButton->setIconSize(QSize(75, 75)); m_nextButton->setVisible(false); m_nextButton->setOpacity(0, false); connect(m_prevButton, &QAbstractButton::clicked, this, &MainWindow::galleryPrev); connect(m_nextButton, &QAbstractButton::clicked, this, &MainWindow::galleryNext); m_am->setupAction(this); m_bottomButtonGroup = new BottomButtonGroup({ m_am->actionActualSize, m_am->actionToggleMaximize, m_am->actionZoomIn, m_am->actionZoomOut, m_am->actionToggleCheckerboard, m_am->actionRotateClockwise }, this); m_bottomButtonGroup->setOpacity(0, false); m_gv->setOpacity(0, false); m_closeButton->setOpacity(0, false); connect(m_pm, &PlaylistManager::loaded, this, [this](int galleryFileCount) { m_prevButton->setVisible(galleryFileCount > 1); m_nextButton->setVisible(galleryFileCount > 1); }); QShortcut * fullscreenShorucut = new QShortcut(QKeySequence(QKeySequence::FullScreen), this); connect(fullscreenShorucut, &QShortcut::activated, this, &MainWindow::toggleFullscreen); centerWindow(); QTimer::singleShot(0, this, [this](){ m_am->setupShortcuts(); }); } MainWindow::~MainWindow() { } void MainWindow::showUrls(const QList &urls) { if (!urls.isEmpty()) { if (urls.count() == 1) { m_graphicsView->showFileFromPath(urls.first().toLocalFile(), true); } else { m_graphicsView->showFileFromPath(urls.first().toLocalFile(), false); m_pm->setPlaylist(urls); m_pm->setCurrentIndex(0); } } else { m_graphicsView->showText(tr("File url list is empty")); return; } m_gv->fitInView(m_gv->sceneRect(), Qt::KeepAspectRatio); } void MainWindow::adjustWindowSizeBySceneRect() { QSize sceneSize = m_graphicsView->sceneRect().toRect().size(); QSize sceneSizeWithMargins = sceneSize + QSize(130, 125); if (m_graphicsView->scaleFactor() < 1 || size().expandedTo(sceneSizeWithMargins) != size()) { // if it scaled down by the resize policy: QSize screenSize = qApp->screenAt(QCursor::pos())->availableSize(); if (screenSize.expandedTo(sceneSize) == screenSize) { // we can show the picture by increase the window size. QSize finalSize = (screenSize.expandedTo(sceneSizeWithMargins) == screenSize) ? sceneSizeWithMargins : screenSize; // We have a very reasonable sizeHint() value ;P this->resize(finalSize.expandedTo(this->sizeHint())); // We're sure the window can display the whole thing with 1:1 scale. // The old window size may cause fitInView call from resize() and the // above resize() call won't reset the scale back to 1:1, so we // just call resetScale() here to ensure the thing is no longer scaled. m_graphicsView->resetScale(); centerWindow(); } else { // toggle maximum showMaximized(); } } } // can be empty if it is NOT from a local file. QUrl MainWindow::currentImageFileUrl() const { QUrl url; std::tie(std::ignore, url) = m_pm->currentFileUrl(); return url; } void MainWindow::clearGallery() { m_pm->clear(); } void MainWindow::loadGalleryBySingleLocalFile(const QString &path) { m_pm->setCurrentFile(path); } void MainWindow::galleryPrev() { int index; QString filePath; std::tie(index, filePath) = m_pm->previousFile(); if (index >= 0) { m_graphicsView->showFileFromPath(filePath, false); m_pm->setCurrentIndex(index); } } void MainWindow::galleryNext() { int index; QString filePath; std::tie(index, filePath) = m_pm->nextFile(); if (index >= 0) { m_graphicsView->showFileFromPath(filePath, false); m_pm->setCurrentIndex(index); } } void MainWindow::showEvent(QShowEvent *event) { updateWidgetsPosition(); return FramelessWindow::showEvent(event); } void MainWindow::enterEvent(QT_ENTER_EVENT *event) { m_bottomButtonGroup->setOpacity(1); m_gv->setOpacity(1); m_closeButton->setOpacity(1); m_prevButton->setOpacity(1); m_nextButton->setOpacity(1); return FramelessWindow::enterEvent(event); } void MainWindow::leaveEvent(QEvent *event) { m_bottomButtonGroup->setOpacity(0); m_gv->setOpacity(0); m_closeButton->setOpacity(0); m_prevButton->setOpacity(0); m_nextButton->setOpacity(0); return FramelessWindow::leaveEvent(event); } void MainWindow::mousePressEvent(QMouseEvent *event) { if (event->buttons() & Qt::LeftButton && !isMaximized()) { m_clickedOnWindow = true; m_oldMousePos = event->pos(); // qDebug() << m_oldMousePos << m_graphicsView->transform().m11() // << m_graphicsView->transform().m22() << m_graphicsView->matrix().m12(); event->accept(); } return FramelessWindow::mousePressEvent(event); } void MainWindow::mouseMoveEvent(QMouseEvent *event) { if (event->buttons() & Qt::LeftButton && m_clickedOnWindow && !isMaximized()) { #if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0) if (!window()->windowHandle()->startSystemMove()) { move(event->globalPos() - m_oldMousePos); } #else move(event->globalPos() - m_oldMousePos); #endif // QT_VERSION >= QT_VERSION_CHECK(5, 15, 0) event->accept(); } return FramelessWindow::mouseMoveEvent(event); } void MainWindow::mouseReleaseEvent(QMouseEvent *event) { m_clickedOnWindow = false; // It seems the forward/back mouse button won't generate a key event [1] so we can't use // QShortcut or QKeySequence to indicate these shortcuts, so we do it here. // Reference: // [1]: https://codereview.qt-project.org/c/qt/qtbase/+/177475 if (event->button() == Qt::ForwardButton || event->button() == Qt::BackButton) { event->button() == Qt::BackButton ? galleryPrev() : galleryNext(); event->accept(); } return FramelessWindow::mouseReleaseEvent(event); } void MainWindow::mouseDoubleClickEvent(QMouseEvent *event) { // The forward/back mouse button can also used to trigger a mouse double-click event // Since we use that for gallery navigation so we ignore these two buttons. if (event->buttons() & Qt::ForwardButton || event->buttons() & Qt::BackButton) { return; } switch (Settings::instance()->doubleClickBehavior()) { case ActionCloseWindow: quitAppAction(); event->accept(); break; case ActionMaximizeWindow: toggleMaximize(); event->accept(); break; case ActionDoNothing: break; } // blumia: don't call parent constructor here, seems it will cause mouse move // event get called even if we set event->accept(); // return QMainWindow::mouseDoubleClickEvent(event); } void MainWindow::wheelEvent(QWheelEvent *event) { QPoint numDegrees = event->angleDelta() / 8; bool needWeelEvent = false, wheelUp = false; bool actionIsZoom = event->modifiers().testFlag(Qt::ControlModifier) || Settings::instance()->mouseWheelBehavior() == ActionZoomImage; // NOTE: Only checking angleDelta since the QWheelEvent::pixelDelta() doc says // pixelDelta() value is driver specific and unreliable on X11... // We are not scrolling the canvas, just zoom in or out, so it probably // doesn't matter here. if (!numDegrees.isNull() && numDegrees.y() != 0) { needWeelEvent = true; wheelUp = numDegrees.y() > 0; } if (needWeelEvent) { if (actionIsZoom) { if (wheelUp) { on_actionZoomIn_triggered(); } else { on_actionZoomOut_triggered(); } } else { if (wheelUp) { galleryPrev(); } else { galleryNext(); } } event->accept(); } else { FramelessWindow::wheelEvent(event); } } void MainWindow::resizeEvent(QResizeEvent *event) { updateWidgetsPosition(); return FramelessWindow::resizeEvent(event); } void MainWindow::contextMenuEvent(QContextMenuEvent *event) { QMenu * menu = new QMenu; QMenu * copyMenu = new QMenu(tr("&Copy")); QUrl currentFileUrl = currentImageFileUrl(); QImage clipboardImage; QUrl clipboardFileUrl; QAction * copyPixmap = m_am->actionCopyPixmap; QAction * copyFilePath = m_am->actionCopyFilePath; copyMenu->addAction(copyPixmap); if (currentFileUrl.isValid()) { copyMenu->addAction(copyFilePath); } QAction * paste = m_am->actionPaste; QAction * stayOnTopMode = m_am->actionToggleStayOnTop; stayOnTopMode->setCheckable(true); stayOnTopMode->setChecked(stayOnTop()); QAction * protectedMode = m_am->actionToggleProtectMode; protectedMode->setCheckable(true); protectedMode->setChecked(m_protectedMode); QAction * toggleSettings = m_am->actionSettings; QAction * helpAction = m_am->actionHelp; QAction * propertiesAction = m_am->actionProperties; #if 0 menu->addAction(m_am->actionOpen); #endif // 0 if (copyMenu->actions().count() == 1) { menu->addActions(copyMenu->actions()); } else { menu->addMenu(copyMenu); } if (canPaste()) { menu->addAction(paste); } menu->addSeparator(); menu->addAction(m_am->actionHorizontalFlip); #if 0 menu->addAction(m_am->actionFitInView); menu->addAction(m_am->actionFitByWidth); #endif // 0 menu->addSeparator(); menu->addAction(stayOnTopMode); menu->addAction(protectedMode); menu->addSeparator(); menu->addAction(toggleSettings); menu->addAction(helpAction); if (currentFileUrl.isValid()) { menu->addSeparator(); menu->addAction(propertiesAction); } menu->exec(mapToGlobal(event->pos())); menu->deleteLater(); copyMenu->deleteLater(); return FramelessWindow::contextMenuEvent(event); } void MainWindow::centerWindow() { this->setGeometry( QStyle::alignedRect( Qt::LeftToRight, Qt::AlignCenter, this->size(), qApp->screenAt(QCursor::pos())->geometry() ) ); } void MainWindow::closeWindow() { QRect windowRect(this->geometry()); m_floatUpAnimation->setStartValue(windowRect); m_floatUpAnimation->setEndValue(windowRect.adjusted(0, -80, 0, 0)); m_floatUpAnimation->setStartValue(QRect(this->geometry().x(), this->geometry().y(), this->geometry().width(), this->geometry().height())); m_floatUpAnimation->setEndValue(QRect(this->geometry().x(), this->geometry().y()-80, this->geometry().width(), this->geometry().height())); m_exitAnimationGroup->start(); } void MainWindow::updateWidgetsPosition() { m_closeButton->move(width() - m_closeButton->width(), 0); m_prevButton->move(25, (height() - m_prevButton->sizeHint().height()) / 2); m_nextButton->move(width() - m_nextButton->sizeHint().width() - 25, (height() - m_prevButton->sizeHint().height()) / 2); m_bottomButtonGroup->move((width() - m_bottomButtonGroup->width()) / 2, height() - m_bottomButtonGroup->height()); m_gv->move(width() - m_gv->width(), height() - m_gv->height()); } void MainWindow::toggleProtectedMode() { m_protectedMode = !m_protectedMode; m_closeButton->setVisible(!m_protectedMode); m_prevButton->setVisible(!m_protectedMode); m_nextButton->setVisible(!m_protectedMode); } void MainWindow::toggleStayOnTop() { setWindowFlag(Qt::WindowStaysOnTopHint, !stayOnTop()); show(); } bool MainWindow::stayOnTop() const { return windowFlags().testFlag(Qt::WindowStaysOnTopHint); } bool MainWindow::canPaste() const { const QMimeData * clipboardData = QApplication::clipboard()->mimeData(); if (clipboardData->hasImage()) { return true; } else if (clipboardData->hasText()) { QString clipboardText(clipboardData->text()); if (clipboardText.startsWith("PICTURE:")) { QString maybeFilename(clipboardText.mid(8)); if (QFile::exists(maybeFilename)) { return true; } } } return false; } void MainWindow::quitAppAction(bool force) { if (!m_protectedMode || force) { closeWindow(); } } void MainWindow::toggleFullscreen() { if (isFullScreen()) { showNormal(); } else { showFullScreen(); } } void MainWindow::toggleMaximize() { if (isMaximized()) { showNormal(); } else { showMaximized(); } } QSize MainWindow::sizeHint() const { return QSize(710, 530); } void MainWindow::on_actionOpen_triggered() { QStringList picturesLocations = QStandardPaths::standardLocations(QStandardPaths::PicturesLocation); QUrl pictureUrl = picturesLocations.isEmpty() ? QUrl::fromLocalFile(picturesLocations.first()) : QUrl::fromLocalFile(QDir::homePath()); QList urls(QFileDialog::getOpenFileUrls(this, QString(), pictureUrl)); if (!urls.isEmpty()) { showUrls(urls); } } void MainWindow::on_actionActualSize_triggered() { m_graphicsView->resetScale(); m_graphicsView->setEnableAutoFitInView(false); } void MainWindow::on_actionToggleMaximize_triggered() { toggleMaximize(); } void MainWindow::on_actionZoomIn_triggered() { if (m_graphicsView->scaleFactor() < 1000) { m_graphicsView->zoomView(1.25); } } void MainWindow::on_actionZoomOut_triggered() { m_graphicsView->zoomView(0.8); } void MainWindow::on_actionHorizontalFlip_triggered() { m_graphicsView->flipView(); } void MainWindow::on_actionFitInView_triggered() { m_graphicsView->fitInView(m_gv->sceneRect(), Qt::KeepAspectRatio); m_graphicsView->setEnableAutoFitInView(m_graphicsView->scaleFactor() <= 1); } void MainWindow::on_actionFitByWidth_triggered() { m_graphicsView->fitByOrientation(); } void MainWindow::on_actionCopyPixmap_triggered() { QClipboard *cb = QApplication::clipboard(); cb->setPixmap(m_graphicsView->scene()->renderToPixmap()); } void MainWindow::on_actionCopyFilePath_triggered() { QUrl currentFileUrl(currentImageFileUrl()); if (currentFileUrl.isValid()) { QClipboard *cb = QApplication::clipboard(); cb->setText(currentFileUrl.toLocalFile()); } } void MainWindow::on_actionPaste_triggered() { QImage clipboardImage; QUrl clipboardFileUrl; const QMimeData * clipboardData = QApplication::clipboard()->mimeData(); if (clipboardData->hasImage()) { QVariant imageVariant(clipboardData->imageData()); if (imageVariant.isValid()) { clipboardImage = qvariant_cast(imageVariant); } } else if (clipboardData->hasText()) { QString clipboardText(clipboardData->text()); if (clipboardText.startsWith("PICTURE:")) { QString maybeFilename(clipboardText.mid(8)); if (QFile::exists(maybeFilename)) { clipboardFileUrl = QUrl::fromLocalFile(maybeFilename); } } } if (!clipboardImage.isNull()) { m_graphicsView->showImage(clipboardImage); clearGallery(); } else if (clipboardFileUrl.isValid()) { QString localFile(clipboardFileUrl.toLocalFile()); m_graphicsView->showFileFromPath(localFile, true); m_pm->setCurrentFile(localFile); } } void MainWindow::on_actionToggleCheckerboard_triggered() { // TODO: is that okay to do this since we plan to support custom shortcuts? m_graphicsView->toggleCheckerboard(QGuiApplication::queryKeyboardModifiers().testFlag(Qt::ShiftModifier)); } void MainWindow::on_actionRotateClockwise_triggered() { m_graphicsView->rotateView(); m_graphicsView->displayScene(); m_gv->setVisible(false); } void MainWindow::on_actionPrevPicture_triggered() { galleryPrev(); } void MainWindow::on_actionNextPicture_triggered() { galleryNext(); } void MainWindow::on_actionToggleStayOnTop_triggered() { toggleStayOnTop(); } void MainWindow::on_actionToggleProtectMode_triggered() { toggleProtectedMode(); } void MainWindow::on_actionSettings_triggered() { SettingsDialog * sd = new SettingsDialog(this); sd->exec(); sd->deleteLater(); } void MainWindow::on_actionHelp_triggered() { AboutDialog * ad = new AboutDialog(this); ad->exec(); ad->deleteLater(); } void MainWindow::on_actionProperties_triggered() { QUrl currentFileUrl = currentImageFileUrl(); if (!currentFileUrl.isValid()) return; MetadataModel * md = new MetadataModel(); md->setFile(currentFileUrl.toLocalFile()); MetadataDialog * ad = new MetadataDialog(this); ad->setMetadataModel(md); ad->exec(); ad->deleteLater(); } void MainWindow::on_actionQuitApp_triggered() { quitAppAction(false); } pineapple-pictures-0.5.3/app/mainwindow.h000066400000000000000000000064731415740431200204500ustar00rootroot00000000000000#ifndef MAINWINDOW_H #define MAINWINDOW_H #include "framelesswindow.h" #include #include #include #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) typedef QEnterEvent QT_ENTER_EVENT; #else typedef QEvent QT_ENTER_EVENT; #endif // QT_VERSION_CHECK(6, 0, 0) QT_BEGIN_NAMESPACE class QGraphicsOpacityEffect; class QGraphicsView; QT_END_NAMESPACE class ActionManager; class PlaylistManager; class ToolButton; class GraphicsView; class NavigatorView; class BottomButtonGroup; class MainWindow : public FramelessWindow { Q_OBJECT public: explicit MainWindow(QWidget *parent = nullptr); ~MainWindow() override; void showUrls(const QList &urls); void adjustWindowSizeBySceneRect(); QUrl currentImageFileUrl() const; void clearGallery(); void loadGalleryBySingleLocalFile(const QString &path); void galleryPrev(); void galleryNext(); protected slots: void showEvent(QShowEvent *event) override; void enterEvent(QT_ENTER_EVENT *event) override; void leaveEvent(QEvent *event) override; void mousePressEvent(QMouseEvent *event) override; void mouseMoveEvent(QMouseEvent *event) override; void mouseReleaseEvent(QMouseEvent *event) override; void mouseDoubleClickEvent(QMouseEvent *event) override; void wheelEvent(QWheelEvent *event) override; void resizeEvent(QResizeEvent *event) override; void contextMenuEvent(QContextMenuEvent *event) override; void centerWindow(); void closeWindow(); void updateWidgetsPosition(); void toggleProtectedMode(); void toggleStayOnTop(); bool stayOnTop() const; bool canPaste() const; void quitAppAction(bool force = false); void toggleFullscreen(); void toggleMaximize(); protected: QSize sizeHint() const override; private slots: void on_actionOpen_triggered(); void on_actionActualSize_triggered(); void on_actionToggleMaximize_triggered(); void on_actionZoomIn_triggered(); void on_actionZoomOut_triggered(); void on_actionToggleCheckerboard_triggered(); void on_actionRotateClockwise_triggered(); void on_actionPrevPicture_triggered(); void on_actionNextPicture_triggered(); void on_actionHorizontalFlip_triggered(); void on_actionFitInView_triggered(); void on_actionFitByWidth_triggered(); void on_actionCopyPixmap_triggered(); void on_actionCopyFilePath_triggered(); void on_actionPaste_triggered(); void on_actionToggleStayOnTop_triggered(); void on_actionToggleProtectMode_triggered(); void on_actionSettings_triggered(); void on_actionHelp_triggered(); void on_actionProperties_triggered(); void on_actionQuitApp_triggered(); private: ActionManager *m_am; PlaylistManager *m_pm; QPoint m_oldMousePos; QPropertyAnimation *m_fadeOutAnimation; QPropertyAnimation *m_floatUpAnimation; QParallelAnimationGroup *m_exitAnimationGroup; ToolButton *m_closeButton; ToolButton *m_prevButton; ToolButton *m_nextButton; GraphicsView *m_graphicsView; NavigatorView *m_gv; BottomButtonGroup *m_bottomButtonGroup; bool m_protectedMode = false; bool m_clickedOnWindow = false; }; #endif // MAINWINDOW_H pineapple-pictures-0.5.3/app/metadatadialog.cpp000066400000000000000000000054571415740431200215700ustar00rootroot00000000000000#include "metadatadialog.h" #include #include #include #include #include #include #include "metadatamodel.h" class PropertyTreeView : public QTreeView { public: explicit PropertyTreeView(QWidget* parent) : QTreeView(parent) {} ~PropertyTreeView() {} protected: void rowsInserted(const QModelIndex& parent, int start, int end) override { QTreeView::rowsInserted(parent, start, end); if (!parent.isValid()) { // we are inserting a section group for (int row = start; row <= end; ++row) { setupSection(row); } } else { // we are inserting a property setRowHidden(parent.row(), QModelIndex(), false); } } void reset() override { QTreeView::reset(); if (model()) { for (int row = 0; row < model()->rowCount(); ++row) { setupSection(row); } } } private: void setupSection(int row) { expand(model()->index(row, 0)); setFirstColumnSpanned(row, QModelIndex(), true); setRowHidden(row, QModelIndex(), !model()->hasChildren(model()->index(row, 0))); } }; class PropertyTreeItemDelegate : public QStyledItemDelegate { public: PropertyTreeItemDelegate(QObject* parent) : QStyledItemDelegate(parent) {} protected: void paint(QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index) const override { QStyleOptionViewItem opt = option; if (!index.parent().isValid()) { opt.font.setBold(true); opt.features.setFlag(QStyleOptionViewItem::Alternate); } QStyledItemDelegate::paint(painter, opt, index); } }; MetadataDialog::MetadataDialog(QWidget *parent) : QDialog(parent) , m_treeView(new PropertyTreeView(this)) { m_treeView->setRootIsDecorated(false); m_treeView->setIndentation(0); m_treeView->setItemDelegate(new PropertyTreeItemDelegate(m_treeView)); m_treeView->header()->resizeSection(0, sizeHint().width() / 2); setWindowTitle(tr("Image Metadata")); QDialogButtonBox * buttonBox = new QDialogButtonBox(QDialogButtonBox::Close); setLayout(new QVBoxLayout); layout()->addWidget(m_treeView); layout()->addWidget(buttonBox); connect(buttonBox, &QDialogButtonBox::rejected, this, &QDialog::close); setWindowFlag(Qt::WindowContextHelpButtonHint, false); } MetadataDialog::~MetadataDialog() { } void MetadataDialog::setMetadataModel(MetadataModel * model) { m_treeView->setModel(model); } QSize MetadataDialog::sizeHint() const { return QSize(520, 350); } pineapple-pictures-0.5.3/app/metadatadialog.h000066400000000000000000000007401415740431200212230ustar00rootroot00000000000000#ifndef METADATADIALOG_H #define METADATADIALOG_H #include QT_BEGIN_NAMESPACE class QTreeView; QT_END_NAMESPACE class MetadataModel; class MetadataDialog : public QDialog { Q_OBJECT public: explicit MetadataDialog(QWidget * parent); ~MetadataDialog() override; void setMetadataModel(MetadataModel * model); QSize sizeHint() const override; private: QTreeView * m_treeView = nullptr; }; #endif // METADATADIALOG_H pineapple-pictures-0.5.3/app/metadatamodel.cpp000066400000000000000000000350531415740431200214240ustar00rootroot00000000000000#include "metadatamodel.h" #include "exiv2wrapper.h" #include #include #include #include MetadataModel::MetadataModel(QObject *parent) : QAbstractItemModel(parent) { } MetadataModel::~MetadataModel() { } void MetadataModel::setFile(const QString &imageFilePath) { QFileInfo fileInfo(imageFilePath); // It'll be fine if we don't re-use the image reader we pass to the graphics scene for now. QImageReader imgReader(imageFilePath); imgReader.setAutoTransform(true); imgReader.setDecideFormatFromContent(true); const QString & itemTypeString = tr("%1 File").arg(QString(imgReader.format().toUpper())); const QString & sizeString = QLocale().formattedDataSize(fileInfo.size()); const QString & birthTimeString = QLocale().toString(fileInfo.birthTime(), QLocale::LongFormat); const QString & lastModifiedTimeString = QLocale().toString(fileInfo.lastModified(), QLocale::LongFormat); const QString & imageDimensionsString = imageSize(imgReader.size()); const QString & imageRatioString = imageSizeRatio(imgReader.size()); appendSection(QStringLiteral("Description"), tr("Description", "Section name.")); appendSection(QStringLiteral("Origin"), tr("Origin", "Section name.")); appendSection(QStringLiteral("Image"), tr("Image", "Section name.")); appendSection(QStringLiteral("Camera"), tr("Camera", "Section name.")); appendSection(QStringLiteral("AdvancedPhoto"), tr("Advanced photo", "Section name.")); appendSection(QStringLiteral("GPS"), tr("GPS", "Section name.")); appendSection(QStringLiteral("File"), tr("File", "Section name.")); if (imgReader.supportsOption(QImageIOHandler::Size)) { appendProperty(QStringLiteral("Image"), QStringLiteral("Image.Dimensions"), tr("Dimensions"), imageDimensionsString); appendProperty(QStringLiteral("Image"), QStringLiteral("Image.SizeRatio"), tr("Aspect ratio"), imageRatioString); } if (imgReader.supportsAnimation() && imgReader.imageCount() > 1) { appendProperty(QStringLiteral("Image"), QStringLiteral("Image.FrameCount"), tr("Frame count"), QString::number(imgReader.imageCount())); } appendProperty(QStringLiteral("File"), QStringLiteral("File.Name"), tr("Name"), fileInfo.fileName()); appendProperty(QStringLiteral("File"), QStringLiteral("File.ItemType"), tr("Item type"), itemTypeString); appendProperty(QStringLiteral("File"), QStringLiteral("File.Path"), tr("Folder path"), fileInfo.path()); appendProperty(QStringLiteral("File"), QStringLiteral("File.Size"), tr("Size"), sizeString); appendProperty(QStringLiteral("File"), QStringLiteral("File.CreatedTime"), tr("Date created"), birthTimeString); appendProperty(QStringLiteral("File"), QStringLiteral("File.LastModified"), tr("Date modified"), lastModifiedTimeString); Exiv2Wrapper wrapper; if (wrapper.load(imageFilePath)) { wrapper.cacheSections(); appendExivPropertyIfExist(wrapper, QStringLiteral("Description"), QStringLiteral("Xmp.dc.title"), tr("Title"), true); appendExivPropertyIfExist(wrapper, QStringLiteral("Description"), QStringLiteral("Exif.Image.ImageDescription"), tr("Subject"), true); appendExivPropertyIfExist(wrapper, QStringLiteral("Description"), QStringLiteral("Exif.Image.Rating"), tr("Rating")); appendExivPropertyIfExist(wrapper, QStringLiteral("Description"), QStringLiteral("Xmp.dc.subject"), tr("Tags")); appendPropertyIfNotEmpty(QStringLiteral("Description"), QStringLiteral("Description.Comments"), tr("Comments"), wrapper.comment()); appendExivPropertyIfExist(wrapper, QStringLiteral("Origin"), QStringLiteral("Exif.Image.Artist"), tr("Authors")); appendExivPropertyIfExist(wrapper, QStringLiteral("Origin"), QStringLiteral("Exif.Photo.DateTimeOriginal"), tr("Date taken")); // FIXME: We may fetch the same type of metadata from different metadata collection. // Current implementation is not pretty and may need to do a rework... // appendExivPropertyIfExist(wrapper, QStringLiteral("Origin"), // QStringLiteral("Xmp.xmp.CreatorTool"), tr("Program name")); appendExivPropertyIfExist(wrapper, QStringLiteral("Origin"), QStringLiteral("Exif.Image.Software"), tr("Program name")); appendExivPropertyIfExist(wrapper, QStringLiteral("Origin"), QStringLiteral("Exif.Image.Copyright"), tr("Copyright")); appendExivPropertyIfExist(wrapper, QStringLiteral("Image"), QStringLiteral("Exif.Image.XResolution"), tr("Horizontal resolution")); appendExivPropertyIfExist(wrapper, QStringLiteral("Image"), QStringLiteral("Exif.Image.YResolution"), tr("Vertical resolution")); appendExivPropertyIfExist(wrapper, QStringLiteral("Image"), QStringLiteral("Exif.Image.ResolutionUnit"), tr("Resolution unit")); appendExivPropertyIfExist(wrapper, QStringLiteral("Image"), QStringLiteral("Exif.Photo.ColorSpace"), tr("Colour representation")); appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"), QStringLiteral("Exif.Image.Make"), tr("Camera maker")); appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"), QStringLiteral("Exif.Image.Model"), tr("Camera model")); appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"), QStringLiteral("Exif.Photo.FNumber"), tr("F-stop")); appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"), QStringLiteral("Exif.Photo.ExposureTime"), tr("Exposure time")); appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"), QStringLiteral("Exif.Photo.ISOSpeedRatings"), tr("ISO speed")); appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"), QStringLiteral("Exif.Photo.ExposureBiasValue"), tr("Exposure bias")); appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"), QStringLiteral("Exif.Photo.FocalLength"), tr("Focal length")); appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"), QStringLiteral("Exif.Photo.MaxApertureValue"), tr("Max aperture")); appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"), QStringLiteral("Exif.Photo.MeteringMode"), tr("Metering mode")); appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"), QStringLiteral("Exif.Photo.Flash"), tr("Flash mode")); appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"), QStringLiteral("Exif.Photo.FocalLengthIn35mmFilm"), tr("35mm focal length")); appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"), QStringLiteral("Exif.Photo.LensModel"), tr("Lens model")); appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"), QStringLiteral("Exif.Photo.BrightnessValue"), tr("Brightness")); appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"), QStringLiteral("Exif.Photo.ExposureProgram"), tr("Exposure program")); appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"), QStringLiteral("Exif.Photo.Saturation"), tr("Saturation")); appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"), QStringLiteral("Exif.Photo.Sharpness"), tr("Sharpness")); appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"), QStringLiteral("Exif.Photo.WhiteBalance"), tr("White balance")); appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"), QStringLiteral("Exif.Photo.DigitalZoomRatio"), tr("Digital zoom")); appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"), QStringLiteral("Exif.Photo.ExifVersion"), tr("EXIF version")); appendExivPropertyIfExist(wrapper, QStringLiteral("GPS"), QStringLiteral("Exif.GPSInfo.GPSLatitudeRef"), tr("Latitude reference")); appendExivPropertyIfExist(wrapper, QStringLiteral("GPS"), QStringLiteral("Exif.GPSInfo.GPSLatitude"), tr("Latitude")); appendExivPropertyIfExist(wrapper, QStringLiteral("GPS"), QStringLiteral("Exif.GPSInfo.GPSLongitudeRef"), tr("Longitude reference")); appendExivPropertyIfExist(wrapper, QStringLiteral("GPS"), QStringLiteral("Exif.GPSInfo.GPSLongitude"), tr("Longitude")); appendExivPropertyIfExist(wrapper, QStringLiteral("GPS"), QStringLiteral("Exif.GPSInfo.GPSAltitudeRef"), tr("Altitude reference")); appendExivPropertyIfExist(wrapper, QStringLiteral("GPS"), QStringLiteral("Exif.GPSInfo.GPSAltitude"), tr("Altitude")); } } QString MetadataModel::imageSize(const QSize &size) { QString imageSize; if (size.isValid()) { imageSize = tr("%1 x %2").arg(QString::number(size.width()), QString::number(size.height())); } else { imageSize = QLatin1Char('-'); } return imageSize; } int simplegcd(int a, int b) { return b == 0 ? a : simplegcd(b, a % b); } QString MetadataModel::imageSizeRatio(const QSize &size) { if (!size.isValid()) { return QStringLiteral("-"); } int gcd = simplegcd(size.width(), size.height()); return tr("%1 : %2").arg(QString::number(size.width() / gcd), QString::number(size.height() / gcd)); } bool MetadataModel::appendSection(const QString §ionKey, QStringView sectionDisplayName) { if (m_sections.contains(sectionKey)) { return false; } m_sections.append(sectionKey); m_sectionProperties[sectionKey] = qMakePair >(sectionDisplayName.toString(), {}); return true; } bool MetadataModel::appendPropertyIfNotEmpty(const QString §ionKey, const QString &propertyKey, const QString &propertyDisplayName, const QString &propertyValue) { if (propertyValue.isEmpty()) return false; return appendProperty(sectionKey, propertyKey, propertyDisplayName, propertyValue); } bool MetadataModel::appendProperty(const QString §ionKey, const QString &propertyKey, QStringView propertyDisplayName, QStringView propertyValue) { if (!m_sections.contains(sectionKey)) { return false; } QList & propertyList = m_sectionProperties[sectionKey].second; if (!propertyList.contains(propertyKey)) { propertyList.append(propertyKey); } m_properties[propertyKey] = qMakePair(propertyDisplayName.toString(), propertyValue.toString()); return true; } bool MetadataModel::appendExivPropertyIfExist(const Exiv2Wrapper &wrapper, const QString §ionKey, const QString &exiv2propertyKey, const QString &propertyDisplayName, bool isXmpString) { const QString & value = wrapper.value(exiv2propertyKey); if (!value.isEmpty()) { appendProperty(sectionKey, exiv2propertyKey, propertyDisplayName.isEmpty() ? wrapper.label(exiv2propertyKey) : propertyDisplayName, isXmpString ? Exiv2Wrapper::XmpValue(value) : value); return true; } return false; } QModelIndex MetadataModel::index(int row, int column, const QModelIndex &parent) const { if (!hasIndex(row, column, parent)) { return QModelIndex(); } if (!parent.isValid()) { return createIndex(row, column, RowType::SectionRow); } else { // internalid param: row means nth section it belongs to. return createIndex(row, column, RowType::PropertyRow + parent.row()); } } QModelIndex MetadataModel::parent(const QModelIndex &child) const { if (!child.isValid()) { return QModelIndex(); } if (child.internalId() == RowType::SectionRow) { return QModelIndex(); } else { return createIndex(child.internalId() - RowType::PropertyRow, 0, SectionRow); } } int MetadataModel::rowCount(const QModelIndex &parent) const { if (!parent.isValid()) { return m_sections.count(); } if (parent.internalId() == RowType::SectionRow) { const QString & sectionKey = m_sections[parent.row()]; return m_sectionProperties[sectionKey].second.count(); } return 0; } int MetadataModel::columnCount(const QModelIndex &) const { // Always key(display name) and value. return 2; } QVariant MetadataModel::data(const QModelIndex &index, int role) const { if (!index.isValid()) { return QVariant(); } if (role != Qt::DisplayRole) { return QVariant(); } if (index.internalId() == RowType::SectionRow) { return (index.column() == 0) ? m_sectionProperties[m_sections[index.row()]].first : QVariant(); } else { int sectionIndex = index.internalId() - RowType::PropertyRow; const QString & sectionKey = m_sections[sectionIndex]; const QList & propertyList = m_sectionProperties[sectionKey].second; return (index.column() == 0) ? m_properties[propertyList[index.row()]].first : m_properties[propertyList[index.row()]].second; } } QVariant MetadataModel::headerData(int section, Qt::Orientation orientation, int role) const { if (orientation == Qt::Vertical || role != Qt::DisplayRole) { return QVariant(); } return section == 0 ? tr("Property") : tr("Value"); } pineapple-pictures-0.5.3/app/metadatamodel.h000066400000000000000000000041241415740431200210640ustar00rootroot00000000000000#ifndef METADATAMODEL_H #define METADATAMODEL_H #include class Exiv2Wrapper; class MetadataModel : public QAbstractItemModel { Q_OBJECT public: explicit MetadataModel(QObject *parent = nullptr); ~MetadataModel(); void setFile(const QString & imageFilePath); static QString imageSize(const QSize &size); static QString imageSizeRatio(const QSize &size); bool appendSection(const QString & sectionKey, QStringView sectionDisplayName); bool appendPropertyIfNotEmpty(const QString & sectionKey, const QString & propertyKey, const QString & propertyDisplayName, const QString & propertyValue = QString()); bool appendProperty(const QString & sectionKey, const QString & propertyKey, QStringView propertyDisplayName, QStringView propertyValue = QString()); bool appendExivPropertyIfExist(const Exiv2Wrapper & wrapper, const QString & sectionKey, const QString & exiv2propertyKey, const QString & propertyDisplayName = QString(), bool isXmpString = false); private: enum RowType : quintptr { SectionRow, PropertyRow, }; QModelIndex index(int row, int column, const QModelIndex &parent = QModelIndex()) const override; QModelIndex parent(const QModelIndex &child) const override; int rowCount(const QModelIndex &parent = QModelIndex()) const override; int columnCount(const QModelIndex & = QModelIndex()) const override; QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override; QVariant headerData(int section, Qt::Orientation orientation, int role = Qt::DisplayRole) const override; // [SECTION_KEY] QList m_sections; // {SECTION_KEY: (SECTION_DISPLAY_NAME, [PROPERTY_KEY])} QMap > > m_sectionProperties; // {PROPERTY_KEY: (PROPERTY_DISPLAY_NAME, PROPERTY_VALUE)} QMap > m_properties; }; #endif // METADATAMODEL_H pineapple-pictures-0.5.3/app/navigatorview.cpp000066400000000000000000000036101415740431200215020ustar00rootroot00000000000000#include "navigatorview.h" #include "graphicsview.h" #include "opacityhelper.h" #include #include NavigatorView::NavigatorView(QWidget *parent) : QGraphicsView (parent) , m_viewportRegion(this->rect()) , m_opacityHelper(new OpacityHelper(this)) { setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setStyleSheet("background-color: rgba(0, 0, 0, 120);" "border-radius: 3px;"); } // doesn't take or manage its ownership void NavigatorView::setMainView(GraphicsView *mainView) { m_mainView = mainView; } void NavigatorView::setOpacity(qreal opacity, bool animated) { m_opacityHelper->setOpacity(opacity, animated); } void NavigatorView::updateMainViewportRegion() { if (m_mainView != nullptr) { m_viewportRegion = mapFromScene(m_mainView->mapToScene(m_mainView->rect())); update(); } } void NavigatorView::mousePressEvent(QMouseEvent *event) { m_mouseDown = true; if (m_mainView) { m_mainView->centerOn(mapToScene(event->pos())); update(); } event->accept(); } void NavigatorView::mouseMoveEvent(QMouseEvent *event) { if (m_mouseDown && m_mainView) { m_mainView->centerOn(mapToScene(event->pos())); update(); event->accept(); } else { event->ignore(); } } void NavigatorView::mouseReleaseEvent(QMouseEvent *event) { m_mouseDown = false; event->accept(); } void NavigatorView::wheelEvent(QWheelEvent *event) { event->ignore(); return QGraphicsView::wheelEvent(event); } void NavigatorView::paintEvent(QPaintEvent *event) { QGraphicsView::paintEvent(event); QPainter painter(viewport()); painter.setPen(QPen(Qt::gray, 2)); painter.drawRect(m_viewportRegion.boundingRect()); } pineapple-pictures-0.5.3/app/navigatorview.h000066400000000000000000000015561415740431200211560ustar00rootroot00000000000000#ifndef NAVIGATORVIEW_H #define NAVIGATORVIEW_H #include class OpacityHelper; class GraphicsView; class NavigatorView : public QGraphicsView { Q_OBJECT public: NavigatorView(QWidget *parent = nullptr); void setMainView(GraphicsView *mainView); void setOpacity(qreal opacity, bool animated = true); public slots: void updateMainViewportRegion(); private: void mousePressEvent(QMouseEvent * event) override; void mouseMoveEvent(QMouseEvent * event) override; void mouseReleaseEvent(QMouseEvent * event) override; void wheelEvent(QWheelEvent *event) override; void paintEvent(QPaintEvent *event) override; bool m_mouseDown = false; QPolygon m_viewportRegion; QGraphicsView *m_mainView = nullptr; OpacityHelper *m_opacityHelper = nullptr; }; #endif // NAVIGATORVIEW_H pineapple-pictures-0.5.3/app/opacityhelper.cpp000066400000000000000000000013341415740431200214660ustar00rootroot00000000000000#include "opacityhelper.h" #include #include OpacityHelper::OpacityHelper(QWidget *parent) : QObject(parent) , m_opacityFx(new QGraphicsOpacityEffect(parent)) , m_opacityAnimation(new QPropertyAnimation(m_opacityFx, "opacity")) { parent->setGraphicsEffect(m_opacityFx); m_opacityAnimation->setDuration(300); } void OpacityHelper::setOpacity(qreal opacity, bool animated) { if (!animated) { m_opacityFx->setOpacity(opacity); return; } m_opacityAnimation->stop(); m_opacityAnimation->setStartValue(m_opacityFx->opacity()); m_opacityAnimation->setEndValue(opacity); m_opacityAnimation->start(); } pineapple-pictures-0.5.3/app/opacityhelper.h000066400000000000000000000007021415740431200211310ustar00rootroot00000000000000#ifndef OPACITYHELPER_H #define OPACITYHELPER_H #include QT_BEGIN_NAMESPACE class QGraphicsOpacityEffect; class QPropertyAnimation; QT_END_NAMESPACE class OpacityHelper : QObject { public: OpacityHelper(QWidget * parent); void setOpacity(qreal opacity, bool animated = true); protected: QGraphicsOpacityEffect * m_opacityFx; QPropertyAnimation * m_opacityAnimation; }; #endif // OPACITYHELPER_H pineapple-pictures-0.5.3/app/playlistmanager.cpp000066400000000000000000000104061415740431200220120ustar00rootroot00000000000000#include "playlistmanager.h" #include #include #include #include PlaylistManager::PlaylistManager(PlaylistType type, QObject *parent) : QObject(parent) , m_type(type) { } PlaylistManager::~PlaylistManager() { } void PlaylistManager::setPlaylistType(PlaylistManager::PlaylistType type) { m_type = type; } PlaylistManager::PlaylistType PlaylistManager::playlistType() const { return m_type; } QStringList PlaylistManager::autoLoadFilterSuffix() const { return m_autoLoadSuffix; } void PlaylistManager::setAutoLoadFilterSuffix(const QStringList & nameFilters) { m_autoLoadSuffix = nameFilters; } void PlaylistManager::clear() { m_currentIndex = -1; m_playlist.clear(); } void PlaylistManager::setPlaylist(const QList &urls) { m_playlist = urls; } void PlaylistManager::setCurrentFile(const QString & filePath) { QFileInfo info(filePath); QDir dir(info.path()); QString && currentFileName = info.fileName(); switch (playlistType()) { case PL_SAMEFOLDER: { if (dir.path() == m_currentDir) { int index = indexOf(filePath); m_currentIndex = index == -1 ? appendFile(filePath) : index; } else { QStringList entryList = dir.entryList( m_autoLoadSuffix, QDir::Files | QDir::NoSymLinks, QDir::NoSort); QCollator collator; collator.setNumericMode(true); std::sort(entryList.begin(), entryList.end(), collator); clear(); int index = -1; for (int i = 0; i < entryList.count(); i++) { const QString & fileName = entryList.at(i); const QString & oneEntry = dir.absoluteFilePath(fileName); const QUrl & url = QUrl::fromLocalFile(oneEntry); m_playlist.append(url); if (fileName == currentFileName) { index = i; } } m_currentIndex = index == -1 ? appendFile(filePath) : index; m_currentDir = dir.path(); } break; } case PL_USERPLAYLIST:{ int index = indexOf(filePath); m_currentIndex = index == -1 ? appendFile(filePath) : index; break; } default: break; } emit loaded(m_playlist.count()); } void PlaylistManager::setCurrentIndex(int index) { if (index < 0 || index >= m_playlist.count()) return; m_currentIndex = index; } int PlaylistManager::appendFile(const QString &filePath) { int index = m_playlist.length(); m_playlist.append(QUrl::fromLocalFile(filePath)); return index; } int PlaylistManager::indexOf(const QString &filePath) { const QUrl & url = QUrl::fromLocalFile(filePath); return m_playlist.indexOf(url); } std::tuple PlaylistManager::previousFile() const { int count = m_playlist.count(); if (count == 0) return std::make_tuple(-1, QString()); int index = m_currentIndex - 1 < 0 ? count - 1 : m_currentIndex - 1; return std::make_tuple(index, m_playlist.at(index).toLocalFile()); } std::tuple PlaylistManager::nextFile() const { int count = m_playlist.count(); if (count == 0) return std::make_tuple(-1, QString()); int index = m_currentIndex + 1 == count ? 0 : m_currentIndex + 1; return std::make_tuple(index, m_playlist.at(index).toLocalFile()); } std::tuple PlaylistManager::currentFile() const { if (m_playlist.count() == 0) return std::make_tuple(-1, QString()); return std::make_tuple(m_currentIndex, m_playlist.at(m_currentIndex).toLocalFile()); } std::tuple PlaylistManager::currentFileUrl() const { if (m_playlist.count() == 0) return std::make_tuple(-1, QUrl()); return std::make_tuple(m_currentIndex, m_playlist.at(m_currentIndex)); } QList PlaylistManager::convertToUrlList(const QStringList &files) { QList urlList; for (const QString & str : qAsConst(files)) { QUrl url = QUrl::fromLocalFile(str); if (url.isValid()) { urlList.append(url); } } return urlList; } pineapple-pictures-0.5.3/app/playlistmanager.h000066400000000000000000000025231415740431200214600ustar00rootroot00000000000000#pragma once #include class PlaylistManager : public QObject { Q_OBJECT public: enum PlaylistType { PL_USERPLAYLIST, // Regular playlist, managed by user. PL_SAMEFOLDER // PlaylistManager managed playlist, loaded from files from same folder. }; explicit PlaylistManager(PlaylistType type = PL_USERPLAYLIST, QObject *parent = nullptr); ~PlaylistManager(); void setPlaylistType(PlaylistType type); PlaylistType playlistType() const; QStringList autoLoadFilterSuffix() const; void setAutoLoadFilterSuffix(const QStringList &nameFilters); void clear(); void setPlaylist(const QList & urls); void setCurrentFile(const QString & filePath); void setCurrentIndex(int index); int appendFile(const QString & filePath); int indexOf(const QString & filePath); std::tuple previousFile() const; std::tuple nextFile() const; std::tuple currentFile() const; std::tuple currentFileUrl() const; static QList convertToUrlList(const QStringList & files); signals: void loaded(int length); private: QList m_playlist; PlaylistType m_type; QString m_currentDir; int m_currentIndex = -1; QStringList m_autoLoadSuffix = {}; }; pineapple-pictures-0.5.3/app/settings.cpp000066400000000000000000000062671415740431200204700ustar00rootroot00000000000000#include "settings.h" #include #include #include #include Settings *Settings::m_settings_instance = nullptr; Settings *Settings::instance() { if (!m_settings_instance) { m_settings_instance = new Settings; } return m_settings_instance; } bool Settings::stayOnTop() { return m_qsettings->value("stay_on_top", true).toBool(); } DoubleClickBehavior Settings::doubleClickBehavior() { QString result = m_qsettings->value("double_click_behavior", "close").toString().toLower(); return stringToDoubleClickBehavior(result); } MouseWheelBehavior Settings::mouseWheelBehavior() { QString result = m_qsettings->value("mouse_wheel_behavior", "close").toString().toLower(); return stringToMouseWheelBehavior(result); } void Settings::setStayOnTop(bool on) { m_qsettings->setValue("stay_on_top", on); m_qsettings->sync(); } void Settings::setDoubleClickBehavior(DoubleClickBehavior dcb) { m_qsettings->setValue("double_click_behavior", doubleClickBehaviorToString(dcb)); m_qsettings->sync(); } void Settings::setMouseWheelBehavior(MouseWheelBehavior mwb) { m_qsettings->setValue("mouse_wheel_behavior", mouseWheelBehaviorToString(mwb)); m_qsettings->sync(); } QString Settings::doubleClickBehaviorToString(DoubleClickBehavior dcb) { static QMap _map { {ActionCloseWindow, "close"}, {ActionMaximizeWindow, "maximize"}, {ActionDoNothing, "ignore"} }; return _map.value(dcb, "close"); } QString Settings::mouseWheelBehaviorToString(MouseWheelBehavior mwb) { static QMap _map { {ActionZoomImage, "zoom"}, {ActionPrevNextImage, "switch"} }; return _map.value(mwb, "zoom"); } DoubleClickBehavior Settings::stringToDoubleClickBehavior(QString str) { static QMap _map { {"close", ActionCloseWindow}, {"maximize", ActionMaximizeWindow}, {"ignore", ActionDoNothing} }; return _map.value(str, ActionCloseWindow); } MouseWheelBehavior Settings::stringToMouseWheelBehavior(QString str) { static QMap _map { {"zoom", ActionZoomImage}, {"switch", ActionPrevNextImage} }; return _map.value(str, ActionZoomImage); } Settings::Settings() : QObject(qApp) { QString configPath; #ifdef FLAG_PORTABLE_MODE_SUPPORT QString portableConfigDirPath = QDir(QCoreApplication::applicationDirPath()).absoluteFilePath("data"); QFileInfo portableConfigDirInfo(portableConfigDirPath); if (portableConfigDirInfo.exists() && portableConfigDirInfo.isDir() && portableConfigDirInfo.isWritable()) { // we can use it. configPath = portableConfigDirPath; } #endif // FLAG_PORTABLE_MODE_SUPPORT // %LOCALAPPDATA% under Windows. if (configPath.isEmpty()) { configPath = QStandardPaths::writableLocation(QStandardPaths::ConfigLocation); } m_qsettings = new QSettings(QDir(configPath).absoluteFilePath("config.ini"), QSettings::IniFormat, this); } pineapple-pictures-0.5.3/app/settings.h000066400000000000000000000022641415740431200201260ustar00rootroot00000000000000#pragma once #include #include enum DoubleClickBehavior { ActionDoNothing, ActionCloseWindow, ActionMaximizeWindow, DCActionStart = ActionDoNothing, DCActionEnd = ActionMaximizeWindow }; enum MouseWheelBehavior { ActionZoomImage, ActionPrevNextImage, MWActionStart = ActionZoomImage, MWActionEnd = ActionPrevNextImage }; class Settings : public QObject { Q_OBJECT public: static Settings *instance(); bool stayOnTop(); DoubleClickBehavior doubleClickBehavior(); MouseWheelBehavior mouseWheelBehavior(); void setStayOnTop(bool on); void setDoubleClickBehavior(DoubleClickBehavior dcb); void setMouseWheelBehavior(MouseWheelBehavior mwb); static QString doubleClickBehaviorToString(DoubleClickBehavior dcb); static QString mouseWheelBehaviorToString(MouseWheelBehavior mwb); static DoubleClickBehavior stringToDoubleClickBehavior(QString str); static MouseWheelBehavior stringToMouseWheelBehavior(QString str); private: Settings(); static Settings *m_settings_instance; QSettings *m_qsettings; signals: public slots: }; pineapple-pictures-0.5.3/app/settingsdialog.cpp000066400000000000000000000053041415740431200216370ustar00rootroot00000000000000#include "settingsdialog.h" #include "settings.h" #include #include #include #include SettingsDialog::SettingsDialog(QWidget *parent) : QDialog(parent) , m_stayOnTop(new QCheckBox) , m_doubleClickBehavior(new QComboBox) , m_mouseWheelBehavior(new QComboBox) { this->setWindowTitle(tr("Settings")); QFormLayout * settingsForm = new QFormLayout(this); static QMap _dc_map { { ActionDoNothing, tr("Do nothing") }, { ActionCloseWindow, tr("Close the window") }, { ActionMaximizeWindow, tr("Toggle maximize") } }; static QMap _mw_map { { ActionZoomImage, tr("Zoom in and out") }, { ActionPrevNextImage, tr("View next or previous item") } }; QStringList dcbDropDown; for (int dcb = DCActionStart; dcb <= DCActionEnd; dcb++) { dcbDropDown.append(_dc_map.value(static_cast(dcb))); } QStringList mwbDropDown; for (int mwb = MWActionStart; mwb <= MWActionEnd; mwb++) { mwbDropDown.append(_mw_map.value(static_cast(mwb))); } settingsForm->addRow(tr("Stay on top when start-up"), m_stayOnTop); settingsForm->addRow(tr("Double-click behavior"), m_doubleClickBehavior); settingsForm->addRow(tr("Mouse wheel behavior"), m_mouseWheelBehavior); m_stayOnTop->setChecked(Settings::instance()->stayOnTop()); m_doubleClickBehavior->setModel(new QStringListModel(dcbDropDown)); DoubleClickBehavior dcb = Settings::instance()->doubleClickBehavior(); m_doubleClickBehavior->setCurrentIndex(static_cast(dcb)); m_mouseWheelBehavior->setModel(new QStringListModel(mwbDropDown)); MouseWheelBehavior mwb = Settings::instance()->mouseWheelBehavior(); m_mouseWheelBehavior->setCurrentIndex(static_cast(mwb)); connect(m_stayOnTop, &QCheckBox::stateChanged, this, [ = ](int state){ Settings::instance()->setStayOnTop(state == Qt::Checked); }); connect(m_doubleClickBehavior, QOverload::of(&QComboBox::currentIndexChanged), this, [ = ](int index){ Settings::instance()->setDoubleClickBehavior(static_cast(index)); }); connect(m_mouseWheelBehavior, QOverload::of(&QComboBox::currentIndexChanged), this, [ = ](int index){ Settings::instance()->setMouseWheelBehavior(static_cast(index)); }); this->setMinimumSize(300, 61); // not sure why it complain "Unable to set geometry" setWindowFlag(Qt::WindowContextHelpButtonHint, false); } SettingsDialog::~SettingsDialog() { } pineapple-pictures-0.5.3/app/settingsdialog.h000066400000000000000000000007541415740431200213100ustar00rootroot00000000000000#ifndef SETTINGSDIALOG_H #define SETTINGSDIALOG_H #include #include class QCheckBox; class QComboBox; class SettingsDialog : public QDialog { Q_OBJECT public: explicit SettingsDialog(QWidget *parent = nullptr); ~SettingsDialog(); signals: public slots: private: QCheckBox * m_stayOnTop = nullptr; QComboBox * m_doubleClickBehavior = nullptr; QComboBox * m_mouseWheelBehavior = nullptr; }; #endif // SETTINGSDIALOG_H pineapple-pictures-0.5.3/app/toolbutton.cpp000066400000000000000000000012471415740431200210320ustar00rootroot00000000000000#include "toolbutton.h" #include "opacityhelper.h" #include #include #include ToolButton::ToolButton(bool hoverColor, QWidget *parent) : QPushButton(parent) , m_opacityHelper(new OpacityHelper(this)) { setFlat(true); QString qss = "QPushButton {" "background: transparent;" "}"; if (hoverColor) { qss += "QPushButton:hover {" "background: red;" "}"; } setStyleSheet(qss); } void ToolButton::setOpacity(qreal opacity, bool animated) { m_opacityHelper->setOpacity(opacity, animated); } pineapple-pictures-0.5.3/app/toolbutton.h000066400000000000000000000006011415740431200204700ustar00rootroot00000000000000#ifndef TOOLBUTTON_H #define TOOLBUTTON_H #include class OpacityHelper; class ToolButton : public QPushButton { Q_OBJECT public: ToolButton(bool hoverColor = false, QWidget * parent = nullptr); public slots: void setOpacity(qreal opacity, bool animated = true); private: OpacityHelper * m_opacityHelper; }; #endif // TOOLBUTTON_H pineapple-pictures-0.5.3/app/translations/000077500000000000000000000000001415740431200206325ustar00rootroot00000000000000pineapple-pictures-0.5.3/app/translations/PineapplePictures.ts000066400000000000000000000606321415740431200246450ustar00rootroot00000000000000 AboutDialog About Launch application with image file path as argument to load the file. Drag and drop image file onto the window is also supported. None of the operations in this application will alter the pictures on disk. Context menu option explanation: Make window stay on top of all other windows. Avoid close window accidentally. (eg. by double clicking the window) Version: %1 Copyright (c) 2020 %1 Logo designed by %1 Built with Qt %1 (%2) Source code Contributors List of contributors on GitHub Thanks to all people who contributed to this project. Translators I would like to thank the following people who volunteered to translate this application. %1 is built on the following free software libraries: Free as in freedom &Special Thanks &Third-party Libraries Your Rights %1 is released under the MIT License. This license grants people a number of freedoms: You are free to use %1, for any purpose You are free to distribute %1 You can study how %1 works and change it You can distribute changed versions of %1 The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. Third-party Libraries used by %1 &Help &About &License GraphicsScene Drag image here GraphicsView File url list is empty File is not a valid image Image data is invalid or currently unsupported Image data is invalid Not supported mimedata: %1 MainWindow File url list is empty &Copy Copy P&ixmap Copy &File Path Properties Stay on top Protected mode Zoom in Zoom out Flip &Horizontally &Paste Toggle Checkerboard Actual size Toggle maximize Rotate right Configure... Help Quit MetadataDialog Image Metadata MetadataModel Origin Section name. Image Section name. File Section name. Camera Section name. %1 File Description Section name. Advanced photo Section name. GPS Section name. Dimensions Aspect ratio Frame count Name Item type Folder path Size Date created Date modified Title Subject Rating Tags Comments Authors Date taken Program name Copyright Horizontal resolution Vertical resolution Resolution unit Colour representation Camera maker Camera model F-stop Exposure time ISO speed Exposure bias Focal length Max aperture Metering mode Flash mode 35mm focal length Lens model Brightness Exposure program Saturation Sharpness White balance Digital zoom EXIF version Latitude reference Latitude Longitude reference Longitude Altitude reference Altitude %1 x %2 %1 : %2 Property Value SettingsDialog Settings Do nothing Close the window Toggle maximize Zoom in and out View next or previous item Stay on top when start-up Double-click behavior Mouse wheel behavior main Pineapple Pictures File list. pineapple-pictures-0.5.3/app/translations/PineapplePictures_de.ts000066400000000000000000000615521415740431200253170ustar00rootroot00000000000000 AboutDialog About Über Launch application with image file path as argument to load the file. Starten Sie die Anwendung mit dem Bilddateipfad als Argument zum Laden der Datei. Drag and drop image file onto the window is also supported. Das Ziehen und Ablegen von Bilddateien in das Fenster wird ebenfalls unterstützt. None of the operations in this application will alter the pictures on disk. Context menu option explanation: Erklärung der Kontextmenüoptionen: Make window stay on top of all other windows. Sicher stellen, dass das Fenster über allen anderen Fenstern bleibt. Avoid close window accidentally. (eg. by double clicking the window) Es vermeiden, das Fenster versehentlich zu schließen. (z.B. durch Doppelklick auf das Fenster) Version: %1 Version: %1 Copyright (c) 2020 %1 Copyright © 2020 %1 Logo designed by %1 Logo entworfen von %1 Built with Qt %1 (%2) Gemacht mit Qt %1 (%2) Source code Quellcode Contributors Mitwirkenden List of contributors on GitHub Liste der Mitwirkenden auf GitHub Thanks to all people who contributed to this project. Vielen Dank an alle, die zu diesem Projekt beigetragen haben. Translators Übersetzer I would like to thank the following people who volunteered to translate this application. Ich möchte den folgenden Personen danken, die sich freiwillig zur Übersetzung dieser Anwendung gemeldet haben. %1 is built on the following free software libraries: Free as in freedom %1 basiert auf den folgenden freien Softwarebibliotheken: &Special Thanks &Besonderer Dank &Third-party Libraries &Bibliotheken von Drittanbietern Your Rights Ihre Rechte %1 is released under the MIT License. %1 wird unter der MIT-Lizenz veröffentlicht. This license grants people a number of freedoms: Diese Lizenz gewährt Menschen eine Reihe von Freiheiten: You are free to use %1, for any purpose Sie dürfen %1 für jeden Zweck verwenden You are free to distribute %1 Sie dürfen %1 verteilen You can study how %1 works and change it Sie können untersuchen, wie %1 funktioniert, und es ändern You can distribute changed versions of %1 Sie können geänderte Versionen von %1 verteilen The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. Die MIT-Lizenz garantiert Ihnen diese Freiheit. Niemand darf es jemals wegnehmen. Third-party Libraries used by %1 Von %1 verwendete Bibliotheken von Drittanbietern &Help &Hilfe &About &Über &License &Lizenz GraphicsScene Drag image here Ziehen Sie das Bild hierher GraphicsView File url list is empty Die Datei-URL-Liste ist leer File is not a valid image Datei ist kein gültiges Bild Image data is invalid or currently unsupported Bilddaten sind ungültig oder werden derzeit nicht unterstützt Image data is invalid Bilddaten sind ungültig Not supported mimedata: %1 Nicht unterstützte Mimedaten: %1 MainWindow File url list is empty Die Datei-URL-Liste ist leer &Copy &Kopieren Copy P&ixmap P&ixmap kopieren Copy &File Path &Dateipfad kopieren Properties Eigenschaften Stay on top Oben bleiben Protected mode Geschützter Modus Zoom in Zoom out Flip &Horizontally &Paste Toggle Checkerboard Actual size Toggle maximize Maximieren umschalten Rotate right Configure... Konfigurieren … Help Hilfe Quit MetadataDialog Image Metadata Bildmetadaten MetadataModel Origin Section name. Ursprung Image Section name. Bild File Section name. Datei Camera Section name. Fotoapparat %1 File %1-Datei Description Section name. Beschreibung Advanced photo Section name. Erweitertes Foto GPS Section name. GPS Dimensions Maße Aspect ratio Seitenverhältnis Frame count Name Name Item type Objekttyp Folder path Ordnerpfad Size Größe Date created Datum erstellt Date modified Datum geändert Title Subject Rating Bewertung Tags Comments Kommentare Authors Autoren Date taken Datum genommen Program name Programmname Copyright Horizontal resolution Horizontale Auflösung Vertical resolution Vertikale Auflösung Resolution unit Auflösungseinheit Colour representation Farbdarstellung Camera maker Kamerahersteller Camera model Kameramodell F-stop Blendenzahl Exposure time Belichtungszeit ISO speed ISO-Geschwindigkeit Exposure bias Belichtungskorrektur Focal length Brennweite Max aperture Maximale Blende Metering mode Messmodus Flash mode Flash-Modus 35mm focal length 35 mm Brennweite Lens model Objektivmodell Brightness Helligkeit Exposure program Belichtungsprogramm Saturation Sättigung Sharpness Schärfe White balance Weißabgleich Digital zoom Digitaler Zoom EXIF version EXIF-Version Latitude reference Latitude Breitengrad Longitude reference Longitude Längengrad Altitude reference Altitude Höhe %1 x %2 %1 × %2 %1 : %2 %1 : %2 Property Eigenschaft Value Wert SettingsDialog Settings Einstellungen Do nothing Nichts tun Close the window Fenster schließen Toggle maximize Maximieren umschalten Zoom in and out View next or previous item Stay on top when start-up Beim Start oben bleiben Double-click behavior Doppelklickverhalten Mouse wheel behavior main Pineapple Pictures Pineapple Pictures File list. Dateiliste. pineapple-pictures-0.5.3/app/translations/PineapplePictures_es.ts000066400000000000000000000616141415740431200253350ustar00rootroot00000000000000 AboutDialog About Acerca de Launch application with image file path as argument to load the file. Lanza la aplicación con la ruta del archivo de imagen como argumento para cargar el archivo. Drag and drop image file onto the window is also supported. También se admite la función de arrastrar y soltar el archivo de imagen en la ventana. None of the operations in this application will alter the pictures on disk. Context menu option explanation: Explicación de la opción del menú contextual: Make window stay on top of all other windows. Haz que la ventana se quede encima de todas las demás ventanas. Avoid close window accidentally. (eg. by double clicking the window) Evita cerrar la ventana accidentalmente. (por ejemplo, haciendo doble clic en la ventana) Version: %1 Versión: %1 Copyright (c) 2020 %1 Derechos reservados (c) 2020 %1 Logo designed by %1 Logo diseñado por %1 Built with Qt %1 (%2) Construido con Qt %1 (%2) Source code Código fuente Contributors Colaboradores List of contributors on GitHub Lista de colaboradores en GitHub Thanks to all people who contributed to this project. Gracias a todas las personas que contibuyen a este proyecto. Translators Traductores I would like to thank the following people who volunteered to translate this application. Me gustaría dar las gracias a las siguientes personas que se ofrecieron a traducir esta solicitud. %1 is built on the following free software libraries: Free as in freedom %1 está construido sobre las siguientes bibliotecas de software libre: &Special Thanks &Agradecimiento especial &Third-party Libraries Your Rights %1 is released under the MIT License. This license grants people a number of freedoms: You are free to use %1, for any purpose You are free to distribute %1 You can study how %1 works and change it You can distribute changed versions of %1 The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. Third-party Libraries used by %1 &Help &About &License GraphicsScene Drag image here GraphicsView File url list is empty File is not a valid image Image data is invalid or currently unsupported Image data is invalid Not supported mimedata: %1 MainWindow File url list is empty &Copy Copy P&ixmap Copy &File Path Properties Stay on top Protected mode Zoom in Zoom out Flip &Horizontally &Paste Toggle Checkerboard Actual size Toggle maximize Rotate right Configure... Help Quit MetadataDialog Image Metadata MetadataModel Origin Section name. Image Section name. File Section name. Camera Section name. %1 File Description Section name. Advanced photo Section name. GPS Section name. Dimensions Aspect ratio Frame count Name Item type Folder path Size Date created Date modified Title Subject Rating Tags Comments Authors Date taken Program name Copyright Horizontal resolution Vertical resolution Resolution unit Colour representation Camera maker Camera model F-stop Exposure time ISO speed Exposure bias Focal length Max aperture Metering mode Flash mode 35mm focal length Lens model Brightness Exposure program Saturation Sharpness White balance Digital zoom EXIF version Latitude reference Latitude Longitude reference Longitude Altitude reference Altitude %1 x %2 %1 : %2 Property Value SettingsDialog Settings Do nothing Close the window Toggle maximize Zoom in and out View next or previous item Stay on top when start-up Double-click behavior Mouse wheel behavior main Pineapple Pictures File list. pineapple-pictures-0.5.3/app/translations/PineapplePictures_fr.ts000066400000000000000000000624601415740431200253350ustar00rootroot00000000000000 AboutDialog About À propos Launch application with image file path as argument to load the file. Lancer l'application avec le chemin du fichier image comme argument pour charger le fichier. Drag and drop image file onto the window is also supported. Le glisser-déposer du fichier image sur la fenêtre est également pris en charge. None of the operations in this application will alter the pictures on disk. Aucun opération dans cette application ne modifiera les fichiers image. Context menu option explanation: Explication des options du menu contextuel : Make window stay on top of all other windows. Faire en sorte que la fenêtre reste au-dessus de toutes les autres fenêtres. Avoid close window accidentally. (eg. by double clicking the window) Éviter de fermer la fenêtre accidentellement. (par exemple en cliquant deux fois sur la fenêtre) Version: %1 Version : %1 Copyright (c) 2020 %1 Copyright © 2020 %1 Logo designed by %1 Logo conçu par %1 Built with Qt %1 (%2) Fait avec Qt %1 (%2) Source code Code source Contributors Contributeurs List of contributors on GitHub Liste des contributeurs sur GitHub Thanks to all people who contributed to this project. Merci à toutes les personnes qui ont contribué à ce projet. Translators Traducteurs I would like to thank the following people who volunteered to translate this application. Je tiens à remercier les personnes suivantes qui se sont portées volontaires pour traduire cette application. %1 is built on the following free software libraries: Free as in freedom %1 est basé sur les bibliothèques de logiciels libres suivantes : &Special Thanks &Remerciement spécial &Third-party Libraries &Bibliothèques tierces Your Rights Vos droits %1 is released under the MIT License. %1 est publié sous licence MIT. This license grants people a number of freedoms: Cette licence accorde aux personnes un certain nombre de libertés : You are free to use %1, for any purpose Vous êtes libre d'utiliser %1, dans n'importe quel but You are free to distribute %1 Vous êtes libre de distribuer %1 You can study how %1 works and change it Vous pouvez étudier le fonctionnement de %1 et le modifier You can distribute changed versions of %1 Vous pouvez distribuer des versions modifiées de %1 The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. La licence MIT vous garantit cette liberté. Personne n'est autorisé à l'enlever. Third-party Libraries used by %1 Bibliothèques tierces utilisées par %1 &Help &Aide &About &À propos &License &Licence GraphicsScene Drag image here Faites glisser l'image ici GraphicsView File url list is empty La liste des URL du fichier est vide File is not a valid image Le fichier n'est pas une image valide Image data is invalid or currently unsupported Les données d'image ne sont pas valides ou ne sont actuellement pas prises en charge Image data is invalid Les données d'image ne sont pas valides Not supported mimedata: %1 Mimedata non pris en charge : %1 MainWindow File url list is empty La liste des URL de fichiers est vide &Copy &Copier Copy P&ixmap Copier P&ixmap Copy &File Path Copier le &chemin du fichier Properties Propriétés Stay on top Rester en-haut Protected mode Mode protégé Zoom in Zoom avant Zoom out Zoom arrière Flip &Horizontally Retourner &horizontalement &Paste Co&ller Toggle Checkerboard Dés/activer le damier Actual size Taille actuelle Toggle maximize Dés/activer l'agrandissement Rotate right Pivoter vers la droite Configure... Configurer… Help Aide Quit Quitter MetadataDialog Image Metadata Métadonnées d'image MetadataModel Origin Section name. Origine Image Section name. Image File Section name. Fichier Camera Section name. Appareil photo %1 File Fichier %1 Description Section name. Description Advanced photo Section name. Photo avancée GPS Section name. GPS Dimensions Dimensions Aspect ratio Rapport d'aspect Frame count Nombre d'images Name Nom Item type Type d'élément Folder path Chemin du dossier Size Taille Date created Date créée Date modified Date modifiée Title Titre Subject Sujet Rating Évaluation Tags Étiquettes Comments Commentaires Authors Auteurs Date taken Date prise Program name Nom du programme Copyright Copyright Horizontal resolution Résolution horizontale Vertical resolution Résolution verticale Resolution unit Unité de résolution Colour representation Représentation des couleurs Camera maker Fabricant de l'appareil photo Camera model Modèle d'appareil photo F-stop Nombre d'ouverture Exposure time Temps d'exposition ISO speed Vitesse ISO Exposure bias Biais d'exposition Focal length Distance focale Max aperture Ouverture maximale Metering mode Mode de mesure Flash mode Mode flash 35mm focal length Distance focale de 35 mm Lens model Modèle d'objectif Brightness Luminosité Exposure program Programme d'exposition Saturation Saturation Sharpness Netteté White balance Balance des blancs Digital zoom Zoom numérique EXIF version Version EXIF Latitude reference Référence de latitude Latitude Latitude Longitude reference Référence de longitude Longitude Longitude Altitude reference Référence d'altitude Altitude Altitude %1 x %2 %1 × %2 %1 : %2 %1 : %2 Property Propriété Value Valeur SettingsDialog Settings Paramètres Do nothing Ne rien faire Close the window Fermer la fenêtre Toggle maximize Activer/désactiver l'agrandissement Zoom in and out Zoom avant et arrière View next or previous item Voir l'élément suivant ou précédent Stay on top when start-up Rester en-haut lors du démarrage Double-click behavior Comportement du double-clic Mouse wheel behavior Comportement de la molette de la souris main Pineapple Pictures Pineapple Pictures File list. Liste des fichiers. pineapple-pictures-0.5.3/app/translations/PineapplePictures_id.ts000066400000000000000000000610751415740431200253230ustar00rootroot00000000000000 AboutDialog About Tentang Launch application with image file path as argument to load the file. Drag and drop image file onto the window is also supported. Tarik dan lepaskan gambar ke jendela juga didukung. None of the operations in this application will alter the pictures on disk. Semua operasi pada aplikasi ini tidak akan mengubah gambar pada diska. Context menu option explanation: Make window stay on top of all other windows. Buat jendela tetap di atas semua jendela lainnya. Avoid close window accidentally. (eg. by double clicking the window) Hindari penutupan jendela secara tidak sengaja (contoh dengan mengklik jendela dua kali) Version: %1 Versi: %1 Copyright (c) 2020 %1 Hak Cipta (c) 2020 %1 Logo designed by %1 Logo didesain oleh %1 Built with Qt %1 (%2) Dibuat dengan Qt %1 (%2) Source code Kode sumber Contributors Kontributor-kontributor List of contributors on GitHub Daftar kontributor di GitHub Thanks to all people who contributed to this project. Terima kasih kepada semua orang yang telah berkontribusi ke proyek ini. Translators Penerjemah I would like to thank the following people who volunteered to translate this application. Saya ingin berterima kasih orang-orang berikut yang secara sukarela menerjemahkan aplikasi ini. %1 is built on the following free software libraries: Free as in freedom &Special Thanks &Third-party Libraries Your Rights %1 is released under the MIT License. %1 diluncurkan di bawah lisensi MIT. This license grants people a number of freedoms: Lisensi ini memberikan orang-orang beberapa kebebasan: You are free to use %1, for any purpose Anda bebas menggunakan %1, untuk tujuan apapun You are free to distribute %1 Anda bebas mendistribusikan %1 You can study how %1 works and change it Anda dapat mempelajari bagaimana cara %1 bekerja dan mengubahnya You can distribute changed versions of %1 Anda dapat mendistribusikan versi %1 yang telah diubah The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. Third-party Libraries used by %1 &Help &Dukungan &About Tentan&g &License &Lisensi GraphicsScene Drag image here Tarik gambar ke sini GraphicsView File url list is empty File is not a valid image Image data is invalid or currently unsupported Image data is invalid Not supported mimedata: %1 MainWindow File url list is empty &Copy &Salin Copy P&ixmap Salin P&ixmap Copy &File Path Salin &Path Berkas Properties Properti Stay on top Tetap di atas Protected mode Mode Terlindungi Zoom in Perbesar Zoom out Perkecil Flip &Horizontally Putar Secara &Horizontal &Paste &Tempel Toggle Checkerboard Actual size Ukuran asli Toggle maximize Rotate right Putar ke kanan Configure... Konfigurasi... Help Dukungan Quit Keluar MetadataDialog Image Metadata Metadata Gambar MetadataModel Origin Section name. Image Section name. Gambar File Section name. Berkas Camera Section name. Kamera %1 File %1 Berkas Description Section name. Keterangan Advanced photo Section name. GPS Section name. GPS Dimensions Dimensi Aspect ratio Frame count Name Nama Item type Jenis item Folder path Path folder Size Ukuran Date created Tanggal dibuat Date modified Tanggal dimodifikasi Title Judul Subject Subyek Rating Tags Tag Comments Komentar Authors Penulis Date taken Tanggal diambil Program name Nama program Copyright Hak cipta Horizontal resolution Resolusi horizontal Vertical resolution Resolusi vertikal Resolution unit Colour representation Representasi warna Camera maker Pembuat kamera Camera model Model kamera F-stop Exposure time ISO speed Exposure bias Focal length Max aperture Metering mode Flash mode 35mm focal length Lens model Model lensa Brightness Kecerahan Exposure program Saturation Sharpness Ketajaman White balance Digital zoom EXIF version Versi EXIF Latitude reference Latitude Longitude reference Longitude Altitude reference Altitude %1 x %2 %1 : %2 Property Properti Value Nilai SettingsDialog Settings Pengaturan Do nothing Jangan lakukan apapun Close the window Tutup jendela Toggle maximize Zoom in and out Perbesar dan perkecil View next or previous item Lihat item berikutnya atau sebelumnya Stay on top when start-up Double-click behavior Mouse wheel behavior main Pineapple Pictures Pineapple Pictures File list. Daftar berkas. pineapple-pictures-0.5.3/app/translations/PineapplePictures_nb_NO.ts000066400000000000000000000612261415740431200257200ustar00rootroot00000000000000 AboutDialog About Om Launch application with image file path as argument to load the file. Kjør programmer ved å angi en filsti som argument for å laste inn filen. Drag and drop image file onto the window is also supported. Å dra og slippe filen i vinduet støttes også. None of the operations in this application will alter the pictures on disk. Ingen av operasjonene i dette programmet vil endre bildet som det er lagret. Context menu option explanation: Forklaring av alternativer i bindeleddsmeny: Make window stay on top of all other windows. Få vinduet til å alltid ligge over andre vinduer. Avoid close window accidentally. (eg. by double clicking the window) Unngå lukking av vinduet ved feiltagelser (f.eks. ved dobbeltklikking av vinduet) Version: %1 Versjon: %1 Copyright (c) 2020 %1 Opphavsrett © 2020 %1 Logo designed by %1 Logo designet av %1 Built with Qt %1 (%2) Bygd med Qt %1 (%2) Source code Kildekode Contributors Bidragsytere List of contributors on GitHub Liste over bidragsytere på GitHub Thanks to all people who contributed to this project. Takk til alle som har bidratt til prosjektet. Translators Oversettere I would like to thank the following people who volunteered to translate this application. Takk til følgende dugnadsoversettere. %1 is built on the following free software libraries: Free as in freedom %1 er bygd med følgende friprog-bibliotek: &Special Thanks &Spesiell takk til &Third-party Libraries &Tredjepartslisenser Your Rights Dine rettigheter %1 is released under the MIT License. %1 er MIT-lisensiert. This license grants people a number of freedoms: Lisensen gir den en rekke friheter: You are free to use %1, for any purpose Du kan bruke %1 som du vil You are free to distribute %1 Du kan dele %1 You can study how %1 works and change it Du kan se kildekoden til %1 og endre den You can distribute changed versions of %1 Du kan distribuere endrede versjoner av %1 The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. MIT-lisensen garanterer deg disse frihetene. Third-party Libraries used by %1 Tredjepartsbibliotek brukt av %1 &Help &Hjelp &About &Om &License &Lisens GraphicsScene Drag image here Dra bilde hit GraphicsView File url list is empty Listen over filnettadresser er tom File is not a valid image Filen er ikke et gyldig bilde Image data is invalid or currently unsupported Ugyldig bildedata, eller for tiden ustøttet Image data is invalid Ugyldig bildedata Not supported mimedata: %1 Ustøttet MIME-data: %1 MainWindow File url list is empty Listen over filnettadresser er ugyldig &Copy &Kopier Copy P&ixmap Kopier p&ixmap Copy &File Path Kopier %filsti Properties Egenskaper Stay on top Behold øverst Protected mode Beskyttet modus Zoom in Førstørr Zoom out Forminsk Flip &Horizontally Vent &vanrett &Paste &Lim inn Toggle Checkerboard Skru av/på rutemønster Actual size Faktisk størrelse Toggle maximize Veksle maksimering Rotate right Roter til høyre Configure... Sett opp … Help Hjelp Quit Avslutt MetadataDialog Image Metadata Bilde-metadata MetadataModel Origin Section name. Opprinnelse Image Section name. Bilde File Section name. Fil Camera Section name. Kamera %1 File %1-fil Description Section name. Beskrivelse Advanced photo Section name. Avansert bilde GPS Section name. GPS Dimensions Dimensjoner Aspect ratio Størrelsesforhold Frame count Rammeantall Name Navn Item type Elementstype Folder path Mappesti Size Størrelse Date created Dato opprettet Date modified Dato endret Title Tittel Subject Emne Rating Vurdering Tags Etiketter Comments Kommentarer Authors Opphavsmenn Date taken Dato tatt Program name Programnavn Copyright Opphavsrett Horizontal resolution Vannrett oppløsning Vertical resolution Loddrett oppløsning Resolution unit Oppløsningsenhet Colour representation Fargerepresentasjon Camera maker Kamerafabrikat Camera model Kameramodell F-stop Blenderåpning Exposure time Eksponeringstid ISO speed ISO-hastighet Exposure bias Eksponeringskorrigering Focal length Brennvidde Max aperture Maks. blenderåpning Metering mode Målingsmodus Flash mode Blitz-modus 35mm focal length 35 mm-brennvidde Lens model Linsemodell Brightness Lysstyrke Exposure program Eksponeringsprogram Saturation Metning Sharpness Skarphet White balance Hvitbalanse Digital zoom Digital forstørrelse EXIF version EXIF-versjon Latitude reference Breddegradsreferanse Latitude Breddegrad Longitude reference Lengdegradsreferanse Longitude Lengdegrad Altitude reference Høydereferanse Altitude Høyde %1 x %2 %1 x %2 %1 : %2 %1 : %2 Property Egenskap Value Verdi SettingsDialog Settings Innstillinger Do nothing Ikke gjør noe Close the window Lukk vinduet Toggle maximize Veksle maksimering Zoom in and out Zoom inn og ut View next or previous item Vis neste eller forrige element Stay on top when start-up Behold i forgrunnen ved oppstart Double-click behavior Dobbeltklikksoppførsel Mouse wheel behavior Musehjulsoppførsel main Pineapple Pictures Ananasbilder File list. Filliste. pineapple-pictures-0.5.3/app/translations/PineapplePictures_nl.ts000066400000000000000000000612541415740431200253370ustar00rootroot00000000000000 AboutDialog About Over Launch application with image file path as argument to load the file. Start het programma met het opgegeven afbeeldingsbestandspad. Drag and drop image file onto the window is also supported. U kunt tevens afbeeldingen naar het venster slepen. None of the operations in this application will alter the pictures on disk. Geen van de handelingen in dit programma veranderen de afbeeldingen op de schijf. Context menu option explanation: Rechtermuisknopmenu-uitleg: Make window stay on top of all other windows. Houdt het venster boven andere vensters. Avoid close window accidentally. (eg. by double clicking the window) Voorkomt per ongeluk sluiten (bijv. door te dubbelklikken op het venster). Version: %1 Versie: %1 Copyright (c) 2020 %1 Copyright (c) 2020 %1 Logo designed by %1 Logo gemaakt door %1 Built with Qt %1 (%2) Gebouwd met Qt %1 (%2) Source code Broncode Contributors Bijdragers List of contributors on GitHub Lijst met bijdragers op GitHub Thanks to all people who contributed to this project. Met dank aan alle personen die hebben bijgedragen aan dit project. Translators Vertalers I would like to thank the following people who volunteered to translate this application. Ik wil graag de volgende mensen bedanken die vrijwillig hebben bijgedragen aan vertalingen. %1 is built on the following free software libraries: Free as in freedom %1 is gebouwd met de volgende vrijesoftwarebibliotheken: &Special Thanks &Met dank aan &Third-party Libraries Ex&terne bibliotheken Your Rights Uw rechten %1 is released under the MIT License. %1 is uitgebracht onder de MIT-licentie. This license grants people a number of freedoms: Deze licentie biedt een hoop vrijheden: You are free to use %1, for any purpose U mag %1 gratis gebruiken, voor welk doeleinde dan ook You are free to distribute %1 U mag %1 vrij verspreiden You can study how %1 works and change it U kunt bekijken hoe %1 werkt en aanpassingen doen You can distribute changed versions of %1 U mag aangepaste versie van %1 vrij verspreiden The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. De MIT-licentie garandeert u deze vrijheid, en niemand mag deze vrijheid wegnemen. Third-party Libraries used by %1 Door %1 gebruikte externe bibliotheken &Help &Hulp &About &Over &License &Licentie GraphicsScene Drag image here Sleep een afbeelding hierheen GraphicsView File url list is empty De bestandspadlijst is leeg File is not a valid image Het bestand is geen afbeelding Image data is invalid or currently unsupported De afbeeldingsgegevens zijn beschadigd of worden niet ondersteund Image data is invalid Beschadigde afbeeldingsgegevens Not supported mimedata: %1 Niet-ondersteunde mime-gegevens: %1 MainWindow File url list is empty De bestandspadlijst is leeg &Copy &Kopiëren Copy P&ixmap P&ixmap kopiëren Copy &File Path &Bestandspad kopiëren Properties Eigenschappen Stay on top Altijd bovenop Protected mode Beschermde modus Zoom in Inzoomen Zoom out Uitzoomen Flip &Horizontally &Horizontaal spiegelen &Paste &Plakken Toggle Checkerboard Schaakbordpatroon aan/uit Actual size Ware grootte Toggle maximize Maximaliseren aan/uit Rotate right Naar rechts draaien Configure... Instellen... Help Hulp Quit Afsluiten MetadataDialog Image Metadata Afbeeldingsmetagegevens MetadataModel Origin Section name. Oorsprong Image Section name. Afbeelding File Section name. Bestand Camera Section name. Camera %1 File %1-bestand Description Section name. Omschrijving Advanced photo Section name. Uitgebreide foto GPS Section name. GPS Dimensions Afmetingen Aspect ratio Beeldverhouding Frame count Aantal frames Name Naam Item type Soort item Folder path Bestandspad Size Grootte Date created Gemaakt op Date modified Bewerkt op Title Naam Subject Onderwerp Rating Waardering Tags Labels Comments Opmerkingen Authors Makers Date taken Genomen op Program name Programmanaam Copyright Copyright Horizontal resolution Horizontale resolutie Vertical resolution Verticale resolutie Resolution unit Resolutie-eenheid Colour representation Kleurweergave Camera maker Camerafabrikant Camera model Cameramodel F-stop Openingsverhouding Exposure time Belichtingstijd ISO speed ISO-snelheid Exposure bias Belichtingsvertekening Focal length Focale lengte Max aperture Max. opening Metering mode Metermodus Flash mode Flitsmodus 35mm focal length 35mm focale lengte Lens model Lensmodel Brightness Helderheid Exposure program Belichtingsprogramma Saturation Verzadiging Sharpness Scherpte White balance Witbalans Digital zoom Digitale zoom EXIF version EXIF-versie Latitude reference Breedtegraadverwijzing Latitude Breedtegraad Longitude reference Lengtegraadverwijzing Longitude Lengtegraad Altitude reference Hoogteverwijzing Altitude Hoogte %1 x %2 %1 x %2 %1 : %2 %1 : %2 Property Eigenschap Value Waarde SettingsDialog Settings Instellingen Do nothing Niets doen Close the window Venster sluiten Toggle maximize Maximaliseren/Demaximaliseren Zoom in and out In-/Uitzoomen View next or previous item Ga naar volgende of vorige item Stay on top when start-up Automatisch altijd bovenop Double-click behavior Dubbelklikgedrag Mouse wheel behavior Scrollwielgedrag main Pineapple Pictures Pineapple Afbeeldingen File list. Bestandslijst. pineapple-pictures-0.5.3/app/translations/PineapplePictures_ru.ts000066400000000000000000000662071415740431200253570ustar00rootroot00000000000000 AboutDialog About О программе Launch application with image file path as argument to load the file. Запустите приложение, указав путь к файлу изображения в качестве аргумента для загрузки файла. Drag and drop image file onto the window is also supported. Также поддерживается перетаскивание файла изображения в окно. None of the operations in this application will alter the pictures on disk. Ни одна из операций в этом приложении не изменит изображения на диске. Context menu option explanation: Пояснение к параметрам контекстного меню: Make window stay on top of all other windows. Расположить окно поверх всех остальных окон. Avoid close window accidentally. (eg. by double clicking the window) Избегать случайного закрытия окна. (например, двойным щелчком по окну) Version: %1 Версия: %1 Copyright (c) 2020 %1 Авторское право (c) 2020 %1 Logo designed by %1 Логотип разработан %1 Built with Qt %1 (%2) Создано с использованием Qt %1 (%2) Source code Исходный код Contributors Участники List of contributors on GitHub Список участников на GitHub Thanks to all people who contributed to this project. Спасибо всем, кто внес свой вклад в этот проект. Translators Переводчики I would like to thank the following people who volunteered to translate this application. Я бы хотел поблагодарить следующих людей, которые приняли участие в переводе этого приложения. %1 is built on the following free software libraries: Free as in freedom %1 создан на следующих бесплатных библиотеках программного обеспечения: &Special Thanks &Особая благодарность &Third-party Libraries &Сторонние библиотеки Your Rights Ваши Права %1 is released under the MIT License. %1 выпущен под лицензией MIT. This license grants people a number of freedoms: Эта лицензия дает людям ряд свобод: You are free to use %1, for any purpose Вы можете свободно использовать %1 для любых целей You are free to distribute %1 Вы можете свободно распространять %1 You can study how %1 works and change it Вы можете изучать, как работает %1, и изменять его You can distribute changed versions of %1 Вы можете распространять измененные версии %1 The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. Лицензия MIT гарантирует вам эту свободу. Никому и никогда не разрешается забирать ее. Third-party Libraries used by %1 Сторонние библиотеки, используемые %1 &Help &Помощь &About &О программе &License &Лицензия GraphicsScene Drag image here Перетащите изображение сюда GraphicsView File url list is empty Список URL-адресов файлов пуст File is not a valid image Файл не является допустимым изображением Image data is invalid or currently unsupported Параметры изображения недействительны или не поддерживаются в настоящее время Image data is invalid Параметры изображения недействительны Not supported mimedata: %1 Неподдерживаемые mimedata: %1 MainWindow File url list is empty Список URL-адресов файлов пуст &Copy &Скопировать Copy P&ixmap Скопировать P&ixmap Copy &File Path Скопировать &путь к файлу Properties Свойства Stay on top Поверх всех окон Protected mode Защищенный режим Zoom in Увеличить Zoom out Уменьшить Flip &Horizontally Отразить по &горизонтали &Paste &Вставить Toggle Checkerboard Переключить фоновый рисунок Actual size Фактический размер Toggle maximize Переключить окно Rotate right Повернуть вправо Configure... Параметры... Help Помощь Quit Выход MetadataDialog Image Metadata Метаданные изображения MetadataModel Origin Section name. Происхождение Image Section name. Изображение File Section name. Файл Camera Section name. Камера %1 File %1 Файл Description Section name. Описание Advanced photo Section name. Расширенное фото GPS Section name. GPS Dimensions Размеры Aspect ratio Соотношение сторон Frame count Количество кадров Name Название Item type Тип элемента Folder path Путь к папке Size Размер Date created Дата создания Date modified Дата изменения Title Заголовок Subject Тема Rating Рейтинг Tags Теги Comments Комментарии Authors Авторы Date taken Дата съемки Program name Название программы Copyright Авторские права Horizontal resolution Разрешение по горизонтали Vertical resolution Разрешение по вертикали Resolution unit Единица разрешения Colour representation Цветопередача Camera maker Производитель камеры Camera model Модель камеры F-stop Величина диафрагмы Exposure time Время экспозиции ISO speed Чувствительность ISO Exposure bias Смещение экспозиции Focal length Фокусное расстояние Max aperture Максимальная апертура Metering mode Режим измерения Flash mode Режим вспышки 35mm focal length Фокусное расстояние 35 мм Lens model Модель объектива Brightness Яркость Exposure program Программа экспозиции Saturation Насыщенность Sharpness Четкость White balance Баланс белого Digital zoom Цифровое увеличение EXIF version Версия EXIF Latitude reference Ссылка на широту Latitude Широта Longitude reference Ссылка на долготу Longitude Долгота Altitude reference Ссылка на высоту Altitude Высота %1 x %2 %1 x %2 %1 : %2 %1 : %2 Property Свойство Value Значение SettingsDialog Settings Параметры Do nothing Ничего не делать Close the window Закрыть окно Toggle maximize Переключить окно Zoom in and out Увеличение и уменьшение масштаба View next or previous item Следующее или предыдущее изображение Stay on top when start-up Поверх всех окон при запуске Double-click behavior Действие при двойном щелчке Mouse wheel behavior Действие колеса мыши main Pineapple Pictures Pineapple Pictures File list. Список файлов. pineapple-pictures-0.5.3/app/translations/PineapplePictures_si.ts000066400000000000000000000637211415740431200253420ustar00rootroot00000000000000 AboutDialog About පිළිබඳව Launch application with image file path as argument to load the file. Drag and drop image file onto the window is also supported. None of the operations in this application will alter the pictures on disk. Context menu option explanation: Make window stay on top of all other windows. Avoid close window accidentally. (eg. by double clicking the window) Version: %1 අනුවාදය: %1 Copyright (c) 2020 %1 ප්‍රකාශන හිමිකම (ඇ) 2020 %1 Logo designed by %1 ලාංඡනය %1 විසින් නිර්මාණය කරන ලදි Built with Qt %1 (%2) Source code Contributors සහදායකයින් List of contributors on GitHub ගිට්හබ් හි සහදායකයින්ගේ ලැයිස්තුව Thanks to all people who contributed to this project. මෙම ව්යාපෘතියට දායක වූ සියලු දෙනාටම ස්තූතියි. Translators පරිවර්තකයින් I would like to thank the following people who volunteered to translate this application. මෙම යෙදුම පරිවර්තනය කිරීමට ස්වේච්ඡාවෙන් ඉදිරිපත් වූ පහත සඳහන් පුද්ගලයින්ට මම ස්තූතිවන්ත වෙමි. %1 is built on the following free software libraries: Free as in freedom &Special Thanks &Third-party Libraries Your Rights ඔබගේ අයිතිවාසිකම් %1 is released under the MIT License. This license grants people a number of freedoms: You are free to use %1, for any purpose ඕනෑම කටයුත්තක් සඳහා %1 භාවිතා කිරීමට ඔබට නිදහස තිබේ You are free to distribute %1 %1 බෙදා හැරීමට ඔබට නිදහස තිබේ You can study how %1 works and change it %1 ක්‍රියා කරන ආකාරය අධ්‍යයනය කර එය වෙනස් කළ හැකිය You can distribute changed versions of %1 %1 හි වෙනස් කළ අනුවාදයන් ඔබට බෙදා හැරීමට හැකිය The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. Third-party Libraries used by %1 &Help උපකාර &About පිළිබඳව &License &බලපත්‍රය GraphicsScene Drag image here GraphicsView File url list is empty ගොනු ඒ.ස.නි. (url) ලැයිස්තුව හිස් ය File is not a valid image ගොනුව වලංගු නොවන රූපයකි Image data is invalid or currently unsupported Image data is invalid රූපයේ දත්ත වලංගු නොවේ Not supported mimedata: %1 MainWindow File url list is empty ගොනු ඒ.ස.නි. (url) ලැයිස්තුව හිස් ය &Copy &පිටපත් Stay on top Protected mode Zoom in Zoom out Flip &Horizontally Copy P&ixmap Copy &File Path &Paste Toggle Checkerboard Actual size Toggle maximize Rotate right Configure... Help Properties Quit MetadataDialog Image Metadata MetadataModel Origin Section name. Image Section name. පින්තූරය File Section name. ගොනුව Camera Section name. %1 File ගොනු %1 Description Section name. විස්තරය Advanced photo Section name. GPS Section name. Dimensions මාන Aspect ratio දර්ශන අනුපාතය Frame count Name නම Item type Folder path ගොනුවේ මාර්ගය Size ප්‍රමාණය Date created සෑදූ දිනය Date modified වෙනස් කළ දිනය Title Subject Rating ශ්‍රේණිගත කිරීම Tags Comments අදහස් Authors කතුවරුන් Date taken ගත් දිනය Program name වැඩසටහනේ නම Copyright Horizontal resolution තිරස් විභේදනය Vertical resolution සිරස් විභේදනය Resolution unit විභේදන ඒකකය Colour representation වර්ණ නිරූපණය Camera maker Camera model F-stop Exposure time නිරාවරණ කාලය ISO speed Exposure bias නිරාවරණ නැඹුරුව Focal length Max aperture Metering mode Flash mode 35mm focal length Lens model කාච ආකෘතිය Brightness දීප්තිය Exposure program නිරාවරණ වැඩසටහන Saturation Sharpness තියුණු බව White balance Digital zoom සංඛ්‍යාංක විශාලනය EXIF version Latitude reference Latitude අක්ෂාංශ Longitude reference Longitude දේශාංශ Altitude reference Altitude උන්නතාංශය %1 x %2 %1 x %2 %1 : %2 %1 : %2 Property Value අගය SettingsDialog Settings සැකසුම් Do nothing කිසිවක් නොකරන්න Close the window කවුළුව වහන්න Toggle maximize Zoom in and out View next or previous item Stay on top when start-up Double-click behavior Mouse wheel behavior main Pineapple Pictures පයින්ඇපල් පික්චර්ස් File list. ගොනු ලැයිස්තුව. pineapple-pictures-0.5.3/app/translations/PineapplePictures_zh_CN.ts000066400000000000000000000607641415740431200257340ustar00rootroot00000000000000 AboutDialog About 关于 Launch application with image file path as argument to load the file. 以图片文件的路径作为参数运行程序即可直接打开图片文件。 Drag and drop image file onto the window is also supported. 也支持拖放图片文件到窗口内来加载图片。 None of the operations in this application will alter the pictures on disk. 此程序中所有的操作均不会修改图片文件本身。 Context menu option explanation: 菜单项说明: Make window stay on top of all other windows. 使窗口始终至于其它非置顶窗口上方。 Avoid close window accidentally. (eg. by double clicking the window) 避免窗口意外关闭。(如:不小心双击了窗口触发了关闭窗口行为) Version: %1 版本: %1 Copyright (c) 2020 %1 版权所有 (c) 2020 %1 Logo designed by %1 Logo 由 %1 设计 Built with Qt %1 (%2) 使用 Qt %1 (%2) 进行构建 Source code 源代码 Contributors 贡献者 List of contributors on GitHub GitHub 上的贡献者列表 Thanks to all people who contributed to this project. 感谢所有参与此项目的朋友。 Translators 翻译者 I would like to thank the following people who volunteered to translate this application. 我想要感谢下列自愿参与翻译此应用程序的朋友。 %1 is built on the following free software libraries: Free as in freedom %1 采用了下列自由软件程序库进行构建: &Special Thanks 致谢(&S) &Third-party Libraries 第三方程序库(&T) Your Rights 用户的权利 %1 is released under the MIT License. %1 是在 MIT 许可协议下发布的。 This license grants people a number of freedoms: 此许可证赋予人们以下自由的权利: You are free to use %1, for any purpose 任何人都可以为了任何目的自由地使用 %1 You are free to distribute %1 任何人都可以自由地分发 %1 You can study how %1 works and change it 任何人都可以自由地研究 %1 的工作原理并对其进行修改 You can distribute changed versions of %1 任何人都可以自由地分发修改过的 %1 版本 The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. 此软件通过 MIT 许可证赋予用户上述自由,任何人无权剥夺。 Third-party Libraries used by %1 %1 使用的第三方程序库 &Help 帮助(&H) &About 关于(&A) &License 软件许可证(&L) GraphicsScene Drag image here 拖放图片至此 GraphicsView File url list is empty 文件 URL 列表为空 File is not a valid image 文件不是有效的图片文件 Image data is invalid or currently unsupported 图像数据无效或暂未支持 Image data is invalid 图片数据无效 Not supported mimedata: %1 不受支持的 MimeData 格式:%1 MainWindow File url list is empty 文件 URL 列表为空 &Copy 复制(&C) Copy P&ixmap 复制位图(&I) Copy &File Path 复制文件路径(&F) Properties 属性 Stay on top 总在最前 Protected mode 保护模式 Zoom in 放大 Zoom out 缩小 Flip &Horizontally 水平翻转(&H) &Paste 粘贴(&P) Toggle Checkerboard 切换棋盘格 Actual size 实际大小 Toggle maximize 最大化窗口 Rotate right 向右旋转 Configure... 设置... Help 帮助 Quit 退出 MetadataDialog Image Metadata 图像元信息 MetadataModel Origin Section name. 来源 Image Section name. 图像 File Section name. 文件 Camera Section name. 照相机 %1 File %1 文件 Description Section name. 说明 Advanced photo Section name. 高级照片 GPS Section name. GPS Dimensions 分辨率 Aspect ratio 纵横比 Frame count 总帧数 Name 名称 Item type 项目类型 Folder path 文件夹路径 Size 大小 Date created 创建日期 Date modified 修改日期 Title 标题 Subject 主题 Rating 分级 Tags 标记 Comments 备注 Authors 作者 Date taken 拍摄日期 Program name 程序名称 Copyright 版权 Horizontal resolution 水平分辨率 Vertical resolution 垂直分辨率 Resolution unit 分辨率单位 Colour representation 颜色表示 Camera maker 照相机制造商 Camera model 照相机型号 F-stop 光圈值 Exposure time 曝光时间 ISO speed ISO 感光度 Exposure bias 曝光补偿 Focal length 焦距 Max aperture 镜头最大光圈 Metering mode 测光模式 Flash mode 闪光灯模式 35mm focal length 换算至 35mm 焦距 Lens model 镜头型号 Brightness 亮度 Exposure program 曝光程序 Saturation 饱和度 Sharpness 清晰度 White balance 白平衡 Digital zoom 数字缩放 EXIF version EXIF 版本 Latitude reference 纬度基准 Latitude 纬度 Longitude reference 经度基准 Longitude 经度 Altitude reference 海拔基准 Altitude 海拔 %1 x %2 %1 x %2 %1 : %2 %1 : %2 Property 属性 Value SettingsDialog Settings 设定 Do nothing 什么也不做 Close the window 关闭窗口 Toggle maximize 最大化窗口 Zoom in and out 放大和缩小 View next or previous item 查看下一项或上一项 Stay on top when start-up 启动时保持窗口总在最前 Double-click behavior 双击时的行为 Mouse wheel behavior 鼠标滚轮行为 main Pineapple Pictures 菠萝看图 File list. 文件列表。 pineapple-pictures-0.5.3/appveyor.yml000066400000000000000000000130421415740431200177210ustar00rootroot00000000000000image: - Visual Studio 2019 environment: CMAKE_INSTALL_ROOT: C:\projects\cmake LIBZ: C:\projects\zlib LIBEXPAT: C:\projects\libexpat LIBAVIF: C:\projects\libavif LIBEXIV2: C:\projects\exiv2 matrix: - build_name: mingw81_64_qt5_15_2 QTPATH: C:\Qt\5.15.2\mingw81_64 MINGW64: C:\Qt\Tools\mingw810_64 install: - mkdir %CMAKE_INSTALL_ROOT% - mkdir %LIBZ% - mkdir %LIBEXPAT% - mkdir %LIBAVIF% - mkdir %LIBEXIV2% - cd %APPVEYOR_BUILD_FOLDER% - git submodule update --init --recursive - set PATH=%PATH%;%CMAKE_INSTALL_ROOT%;%QTPATH%\bin;%MINGW64%\bin - set CC=%MINGW64%\bin\gcc.exe - set CXX=%MINGW64%\bin\g++.exe build_script: # prepare - mkdir 3rdparty - cinst ninja # download and install zlib for KArchive - cd %LIBZ% - curl -fsSL -o zlib1211.zip https://zlib.net/zlib1211.zip - 7z x zlib1211.zip -y - cd zlib-1.2.11 - mkdir build - cd build - cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_ROOT% - cmake --build . --config Release - cmake --build . --config Release --target install/strip - cd %APPVEYOR_BUILD_FOLDER% # install ECM so we can build KImageFormats - cd 3rdparty - git clone -q https://invent.kde.org/frameworks/extra-cmake-modules.git - git rev-parse HEAD - cd extra-cmake-modules - cmake -G "Ninja" . -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_ROOT% -DBUILD_TESTING=OFF - cmake --build . - cmake --build . --target install - cd %APPVEYOR_BUILD_FOLDER% # install AOM for libavif AV1 decoding support... - cd 3rdparty - git clone -b v3.1.2 --depth 1 https://aomedia.googlesource.com/aom - cd aom - mkdir build.libavif - cd build.libavif - cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_ROOT% -DENABLE_DOCS=OFF -DBUILD_SHARED_LIBS=ON -DAOM_TARGET_CPU=generic -DENABLE_TESTS=OFF -DENABLE_TESTDATA=OFF -DENABLE_TOOLS=OFF -DENABLE_EXAMPLES=0 - cmake --build . --config Release - cmake --build . --config Release --target install/strip - cd %APPVEYOR_BUILD_FOLDER% # install libavif for avif format support of KImageFormats - cd %LIBAVIF% - curl -fsSL -o libavif-v0_9_0.zip https://github.com/AOMediaCodec/libavif/archive/v0.9.0.zip - 7z x libavif-v0_9_0.zip -y - cd libavif-0.9.0 - mkdir build - cd build - cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_ROOT% -DAVIF_CODEC_AOM=ON - cmake --build . --config Release - cmake --build . --config Release --target install/strip - cd %APPVEYOR_BUILD_FOLDER% # install KArchive for kra format support of KImageFormats - cd 3rdparty - git clone -q https://invent.kde.org/frameworks/karchive.git - git rev-parse HEAD - cd karchive - mkdir build - cd build - cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_ROOT% - cmake --build . --config Release - cmake --build . --config Release --target install/strip - cd %APPVEYOR_BUILD_FOLDER% # build libexpat for libexiv2 - cd %LIBEXPAT% - curl -fsSL -o R_2_4_1.zip https://github.com/libexpat/libexpat/archive/R_2_4_1.zip - 7z x R_2_4_1.zip -y - cd libexpat-R_2_4_1/expat/ - cmake -G "Ninja" . -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_ROOT% -DEXPAT_BUILD_EXAMPLES=OFF -DEXPAT_BUILD_TESTS=OFF -DEXPAT_BUILD_TOOLS=OFF - cmake --build . --target install/strip - cd %APPVEYOR_BUILD_FOLDER% # build libexiv2 - cd %LIBEXIV2% - curl -fsSL -o v0.27.4.zip https://github.com/Exiv2/exiv2/archive/v0.27.4.zip - 7z x v0.27.4.zip -y - cd exiv2-0.27.4 - cmake -G "Ninja" . -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_ROOT% -DEXIV2_BUILD_SAMPLES=OFF -DEXIV2_ENABLE_WIN_UNICODE=ON - cmake --build . --target install/strip - cd %APPVEYOR_BUILD_FOLDER% # install KImageFormats - cd 3rdparty - git clone -q https://invent.kde.org/frameworks/kimageformats.git - git rev-parse HEAD - cd kimageformats - mkdir build - cd build - cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DKDE_INSTALL_QTPLUGINDIR=%QTPATH%\plugins - cmake --build . --config Release - cmake --build . --config Release --target install/strip - cd %APPVEYOR_BUILD_FOLDER% # finally... - mkdir build - cd build - cmake .. -G "Unix Makefiles" -DCMAKE_BUILD_TYPE=Release -DCMAKE_MAKE_PROGRAM=mingw32-make -DCMAKE_INSTALL_PREFIX='%cd%' - cmake --build . --config Release - cmake --build . --config Release --target install/strip # fixme: I don't know how to NOT make the binary installed to the ./bin/ folder... - cd bin - copy %APPVEYOR_BUILD_FOLDER%\LICENSE . - copy %CMAKE_INSTALL_ROOT%\lib\libaom.dll - copy %CMAKE_INSTALL_ROOT%\bin\libexpat.dll - copy %CMAKE_INSTALL_ROOT%\bin\libexiv2.dll - copy %CMAKE_INSTALL_ROOT%\bin\libavif.dll - copy %CMAKE_INSTALL_ROOT%\bin\libzlib.dll - copy C:\projects\cmake\bin\libKF5Archive.dll . - windeployqt --verbose=2 --no-quick-import --no-translations --no-opengl-sw --no-angle --no-system-d3d-compiler .\ppic.exe # copy 3rdparty licenses for the libs we vendored for windows... - mkdir licenses - cd licenses - copy %APPVEYOR_BUILD_FOLDER%\3rdparty\aom\LICENSE License.aom.txt - copy %APPVEYOR_BUILD_FOLDER%\3rdparty\karchive\LICENSES\LGPL-2.0-or-later.txt License.KArchive.txt - copy %APPVEYOR_BUILD_FOLDER%\3rdparty\kimageformats\LICENSES\LGPL-2.1-or-later.txt License.kimageformats.txt - copy %LIBEXPAT%\libexpat-R_2_4_1\expat\COPYING License.expat.txt - copy %LIBAVIF%\libavif-0.9.0\LICENSE License.libavif.txt - copy %LIBEXIV2%\exiv2-0.27.4\COPYING License.exiv2.txt # TODO: Qt, zlib - cd .. # for debug.. - tree /f artifacts: - path: build\bin pineapple-pictures-0.5.3/assets/000077500000000000000000000000001415740431200166335ustar00rootroot00000000000000pineapple-pictures-0.5.3/assets/icons/000077500000000000000000000000001415740431200177465ustar00rootroot00000000000000pineapple-pictures-0.5.3/assets/icons/app-icon.ico000066400000000000000000006474121415740431200221660ustar00rootroot00000000000000 hf  00 %v@@ (B; (F} n(  wwwLww4wwww7w٤wwqwȉvx)w˥wwwwtqv!wexyx|pgnQlenympwr☁u4|tݱvuwb{٪vvJOh@Hf?HfAIf?Hf@Hf @Hfwmxk}xINi>Fd@Hf?Gf@Hf@Hf@Hf`@Hfww2{䰘gi|JRn@HfAIhYe}\\o?Hf@Hf@Hf@Hf~xvyip`g[b|kx|KPi?Hf@Hf@Hft z|rmb`p>Ge@Hfv@HftzȰҤgfl۫[ߧ\۞X}Y^t@Hf=Ed>Fdsyͤp:˧^ťhmrԝS{qFMjQZvhq{v{xH*,mz^NXu@HfCLjWfuq}wv*{ݮ£|ޫF38HǢuutxgqWoW~Kvww]|£zģyz[ cvuwPzȩ|楊wvvtsvw[xxwuv4uu( @ xwwwcwMw wxwwwwwwwwwwwwwwwउw www wwwwww̤wwwwwzwwwww礉wKwwxywfwwwwww_wwwwww0wEwLwCw-wwysvSv뛃vvwwwrwwvww wFwwҤxxxxx릊w̖uPRitWWjSTiPSiXXjlenrqrq qqwww5wxz|~~~wZYjAIf?Gf?Gf?Gf?Gf?Gf?Hf@Hf@Hf@Hf@Iewww^w⦋z~~ORj>Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hfwwwgx{RUl?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hfw@Hg@HfwwwLx맍|{xpo}QUm?Gf@Hf@Hf?Gf>Ge>Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf @Hfwww˦{}y[`xRYuJRoAIg?Gf@Hf@Hf?GeKQkdbs`_qGMh?Gf@Hf@Hf@Hf@Hf@Hf@HfV@HfwwwyyvvYa{IQmAIf?Ge@Hf@Hf?GeBIgRcy}KPi?Gf@Hf@Hf@Hf@Hf@Hf{@Hf@HfwwwЧ|~fmioflX_yJRnEMjGOlQXsepyyuxAIf@Hf@Hf@Hf@Hf@Hf@Hf@HfwwKxntoupvrxsyqwouqwx}qUWl?Gf@Hf@Hf@Hf@Hfh@Hf@Hgwww~ytzw}x~z{|~FQqBIfmjvqks>Gf@Hf@Hf@Hf@Hf7@HfwvwzȜ`~wYayy@Hf@Hf@Hf@Hf@Hf @Hfwvw{äGŝ_şcKdzήLLXv{AHf@Hf@Hf@Hf>@Hf@Hfwww{äH=Šau:;@DIL۟\}IRo`hHQn@He@Hfd@If@HfwwzM<6̦Zy;JQADHҜ\}JRo?GePYuhrej~RZuU_{wwwuyڦ]<6.ڬGԫU4F]۠Lp~NXv?Gf@Hf@HfV_{s~vr~*~fsww@x~ȥ{@5.(+/KaӝROYw@Hf@Hf@Hf@Hf@HfQZulxtQygrwww§{֧a5.*-15W_BDZOZx?Ge?Ge>Gf?Gf?GeFXRjfozwwwexةY/(,157;@G̤tkJcbaqmfoc`mMc[`-_wwwzΨi=/038@ަS˥xxyya`^^www4wۦ{ȧvШgҨeϧmƥ}zwɥu"R _&^^wwwIwߦz~ywѤw6wwwxw>wȥy||xww.wwwwwwwॊy{|~~}|zxwؤwtwwuwww%wlwwѤx㤊x礉wᤉwΤwwcwwwwwwww(w.w'wwwt?(0` $wwww(w wvwwwtwؤw⤉ww6wwwwwuwwwwwӤw(wwxw`wwwwwww|wtwwwMw褉wwwwwwwxswwwGf@Hf@Hf@Hf@Hf@Hf@Hf@Hf(@Hfww@wzz}tzu{v|w|x}x~yz{|}~}GRr>Fdʠ[l:9ˣ[{<994ЧT=58;>BDGIL̙`PZw?Ge?GeIRomxcm^_qRQdRTiww[w{˥zD>94.تIçq92Hҙ֥[@DFOmWb@Hf@Hf@Hf?GeQZvr~ztNkwp|ww4wyN>950*5>013әߺEIfw]hBJg@Hf@Hf@Hf@Hf@HfS\xq}vtr~Mkvlwwwwʥx}Ӧi=950+)+/13٤FޡJr^iBKh@Hf@Hf@Hf@Hf@Hf@Hf@HfNXtkvvsiulxxvww{L850+),/13Vn@D٠RvqBJg@Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GdHQocnvs~]jww:wy~ɦz?40+),/146=B>BDFʗYWf@Ii@He?Hf?Gf?Gf?Gf@Hf@GdETzRuOsOqwwwx{ͧq;.+),/1479_wwwwz˨pժ\۪QݪOܩSר^Φpå|xwउw9_ _D^ ^www+wͤxz|xw줉wWwwwww3wͤwz}~{xw꤉w`wwvww*wwy{~|yww٤wPwwuwwww꤉xy{}~|zxwww.wwwww;wwwxy{|||||{zyxww¤w[w wxwww9wwʤwwwxxxwwwwؤwwPwvxwwww3wYwzwwwwwew@www??|?????(@ @wwwxwwxw3wwwwLwwwwwGwޤwwwwwwwwww7w٤wwwwwwwwuwzww)wʤwwwwwwwwդwwwwwwwwwwwwww契w3wwwwwwwwwwwwww⤉w$www wwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwkwwwwwwwwwww̤w*wtwvwXwwwwwwwwwwwڤw9wwwwwEw㤉wwwwwwwwww椉wJxvvww5wפwwwwwwwwww契w]{wzwwww7wUwowwww|whwMw.wwvvx&wȢwvvvwwwwwwwqvv~www*wjwwפwwwwwwwwww뤉wͤwwYxxVWj%vkonfn^\kUVjSTiWWj`]knfnsquwwwwxww wGww変wwwxxyyyyyxxxwww~tWWjAHfBIf?Gf>Gf?Gf?Gf?Gf>Gf?GfAIfNQhc_lVVj=)):cwww>wwwwxy{||}}~~}}|{zx|opORiAIf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?Gf?Gf@Hf@Hfy@Hf@Hf@Ievwwww뤉wxy{|~}leoBJf?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf&@Hf?Hfv|w6wwwy{}ebp@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf+@HfAIhwvwMwܤwxz}gdr?Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfwywVw椉wx{~miu@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfxvwLw椉wy|snyAHf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfM@HfBHevww5wڤwy|olzBIg@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf @Hfwwwwx|oo`d{Y`yX_yT[vHOl?Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf>@Hfwvwwx{dh}X`zX`zZa{X_yQXsEMj@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?Ge>Ge@Hf@Hf>Gf?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Heww;w餉wz~hj}OVrQYtSZuQXtLTpFMjAIf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@GeAIfVYnvq{}z}mhtLPi?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hfwwwwy|ej[b|LTpAIg@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GeFQs`u]\n?Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfwwBwx{ekahciahSZuCKi?Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GeBIgR_ks~[[m?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf"@Hfwwwwy}kpdkflfmgmgn_fNUqBJh?Ge?Gf@Hf@Hf@Hf?Ge?GeBJhNUqbjvzwt~LPi?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf&@Hfww%wउwzxygniojpjpkqlrmslrbiT\vJRnEMjDLiEMjJRnU\wekqwy~{yvwnu@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf @Hfswwawx|kqlrmsmtntououpvqwrxrxpvlrkqmsrxv|w}y~{OSj?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hfwwwwy~y{oupvqwqwrxsysytzuzu{v|w}x}y~yzz{hWbmfntozohr?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfwwwϤwzrxsytzuzu{v|w|w}x~y~yz{{|}}~}HTt>Fd>Ff>FeJPj~{CJg@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hfu@Hf?Heww.w뤊x{|y}w}x}y~yzz{||}~~yAJh?Gf>Gf?Gf@HffdtOSj?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf5@HfwwJwx|ޤRvz||}~~aZ`IMei^`xg^]n>Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hfwwbwx}ƥGܢL~ҫiƫKҗOĎSݛQ^jeq>Gf@Hf@Hf@Hf@Hf@Hf@Hf@HfJ@HfEFbwwrwy}ʦ{FBաQգQ9ެZ̭JJLNPYnhr>Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hfwwzwy}̦wEC>СUơc9<Ge@Hf@Hf@Hf@Hf@Hf@Hf&@Hfwwxwy}̦xEC?:΢Vt::<>??CFHJLNPW~GOmGOmqxV]v?Ge@Hf@Hf@Hf@Hf@HfF@Hf?Iewwmwy}ɦ}FC?;7ФT=7:<>@BDFHJLNOϚbOYv?Ge@HfYc~o{R[w@Hf?Gf@Hf@HfQ=I^@HewwYwx}ĥJB?;83ԧO?588:=@BDFHJLNėeYc@Hf@Hf@HfDLjhssXa}VWkWViFŝ|>Geww@wx|SB?;84/۪E=35;ZbF?BDFHIPp~alBKh@Hf@Hf@Hf?GeKTpozu}䃂2irww#w⤉w{צeA?;840+:yϪ`5149ҕݶMADEJș^wgrFNk@Gf@Hf@Hf@Hf@Hf@GfPYuq|vtr~cmxT]zww wwz~ɦ~E?;840-(+/,/22Xł?FØ^}|ujuHQn?Ge@Hf@Hf@Hf@Hf@Hf@Hf@HfPYvo{wvtr~qmyp{wwwwy}ݦW>;840-))+.021`Ɍ?FsuvkvJSo?Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfLUrjuvvtnz)p|wwHwx{ʦ{B;840-)*,.023?[@DJxgrHQn?Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GeFNk^hr}r}Ecpwwwʤwz~קa:840-)*,.0245GȁˌY?BDFLs_AIf@Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@GeAJjETyXa|0iswwwxwx|ިQ640-)*,.0246779<@BDFGҝ\_uBNq@Ge@Hf@Hf?Hf@Hf@Hf@Hf@Hf@Hf@GeBNpSv\t\]ww%wۤwz~æK30-)*,.02468:<>@BDFG΢ooU}DRw?GdAIfKOiFLgDJgFLgCJf?GdCPtU|`^P_wvw|wx{¦ݪO0,)*,.02468:<>@BCHեiw_YH_bbt}yzu|txnrK`X``]+]wwwʤwy}ԩa5(),.02468:<>?AQ̥x__m|xwz```_\\xwwQwwz}ŧجS4+,.13579;?MҦkfr}ywwԦt&U ``_eZwwwwxz~§ϩj٫SF?>@FިQ֧b˦x}yww契wTwd^ ^_^,^wwwwx{~}zwwwywwANdANdANdANdvwwwwxz}|ywwww wxww!wwxz|~|ywwww wvwwwwwy{}}zxww契wxw www wuw褉wxy{}~}{ywwwӤwQwwvvwwܤwwwwwwwwwwwwwwwwwwwwwww wwww/wΤwwwwwwwwwwwwwwwwwwwwwwvvww!wwwwwwwwwwwwwwwwwwwwwwwwxswwwwwwwwwwwwwwwwwwwwwwwwwȤw)xsvwwwwwwwwwwwwwwwwwwwwwwwwդw6wxww wwwwwwwwwwwwwwwwwwwwwwwᤉwGuxuvw www$w'w*w(w%w!wwvstvvwrwwwwwwwwwwwwwwwwwwwwww줉wYetz{w w"wCwfwwwwͤwڤw䤉w餉w줉w뤉w礉wउwӤwƤwwwwwRw1vvsywvw_wwwwwvvvwwwwwwwwwwwwwwwlwwwwwwKwwwޤwwwwwwwwwwwwwwwwwwww餉w̤wwiw2w oy{4bwMv嗀uxrznpnfne`l^[kZYk[Zk`]licmrio~ppysuvwwwwwwwvvxyxwWwwؤwwwwwwwwwwwwwwwwwwwwwwwwwwww꤉wwzw5x vBKh4Bd][kMmen][kMPhDKg@Hf?Gf>Gf?Gf?Gf?Gf?Gf>Gf?GfAIfEKgNQh][ktjozsvwwww wwww8wwӤwwwwwwwwwwwwxxxxxxxxxxwwwwwwwwwwwwꦊwu]GMgR@HfBIf@Hf?Gf?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?Gf?Gf?HfFLgZYj{opuwuvw wGww꤉wwwwwwwwxxxyyyzzzzzzzzzyyyxxxwwwwwwvtq[ZkBIf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GfAIfCJg?Gfo@Hf@Hh@HexwwAwwwwwwwwwxyyzz{{||||}}}}}||||{{{zyyxxw{sa^lDKg?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfRAGe@GfGUUxzw,wwwwwwwwxyyz{{||}}~~~~~~~~~}}}||{{yynqKOh?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf AIc@Hhwxwmwۤwwwwwwxyz{||}~~~~zicnBJg?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfAAFc@Hevsw1wwwwwwwxyz{|}~|_]m@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf^>Gf?Hfxxw`wܤwwwwwxyz{|}~}XXl?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hfn@Gf@GfOywwwwwwwxyz{|~VXl?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfnAJfAIfbuw wwwwwwxz{|}VXm?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf]EBcAGewww-wwwwwxyz|}~Z[n?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfAAHe@Hfwvw4wʤwwwwxy{|~_^p?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hewww3wͤwwwwxz{}~dbs?Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfAIe@Heuxw,wȤwwwwxz|}hgu@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfH@Hf?Igxww wwwwwyz|}nkx@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hfwwwwwwwyz|}olyAHf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfU?GfDJdvvwwwwwxz|}{tsnokl~jk~klkl~\^sAIg@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfAGf AGft{w[wwwwxz|}rqae{X_xU\wU\wU\wU]xV]xV^xSZuEMk?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf>@Hfvww.wפwwwxz{}ts^czV]xU\wV]xW^xW^yX_yX_yY`zY`zRYtELj@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfAJ[@Flww wwwwxy{}mnY`yV]xW^yX_yX_zY`zY`zZ`zZa{Za{W^yLTpBJh@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hfwywewwwwy{}~srZazX_yY`zY`zZa{Za{[a{[b|\b|[b|X_yPWsEMj@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?Gf>Ge>Ge>Ge>Ge?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf;@Hfww'w֤wwwxz|~]bzW_yY`zZa{[b|[b|\b|[b|[b|X_yS[vLTpELj@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?Gf?GeELhSVm__qgesecs[\oLQjAIf?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hfk?HfCEdtswwwwxz|}utZa{KRoFMjIPmKRoLTpLTpLSoJQnFNkCKiAIg@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@He@GeNRkpmywzYZmAIf?Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfCEgAGfww;w餉wwwy{}kn\c}]d~Y`zGNk?Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GeCNn^lyqwINh?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?If ?Ifwxwwwwxz|~gk^e~_f_f`g[b|JRn@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Ge@IgM]eu}{NRj?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfAHfAHfuww=w줉wwwy{}gl_fagagahbhbi_fPWsBJg@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GeFPp\tnsrs|MQi?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf$@Hfvvwwwwxz|~knahbicicicjdjdjdkdjX_yFNk@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GfBJhQYtjvutrrsyyFLh?Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf-@Hfww.w㤉wwwy{}qtbidjdkdkekelelflflgmgmbhQYtCKi?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GfAIgNUqahq}yxxvutsstltAHf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf1@Hfxuwzwwwxz|~}|dkelflfmfmgmgmgnhnhnhoioipio`fOVrCKh?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GeBJgMTq`fntryy|zzxwvvtxZZm?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf/@HfwwwǤwwwy{}gmgmgnhnhnhoioioipjpjpjqkqkqkrlrkqbhRZuFNkAIf?Ge?Ge@Hf@Hf@Hf@Hf@Hf@Hf?Ge?Ge@HfFMjRYtciousysyx~}|zyxxwu|zELg@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf)@HfwwGwwwxz|~oshoioipjpjpjqkqkqkrlrlrlrmsmsmsntntountio^e~SZuKSoFMkCKhBJhBJhCKhFMjKRoS[u_fkqsxuztzuzv}|{zyxveao?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hfxswwwwx{}ipkqkqkqlrlrlrmsmsmsntntntouououpvpvpvqwqwrxqwoukrhoflfmiolsqvtzv{v{v{v|v|w|~~}|{zx{{FLh@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?If?Ifwwwˤwwwy{~ntlrmsmsmsntntntouououpvpvpvqwqwqwrwrxrxsxsysytytzuzu{v{v|v|v|v|w|w}w}w}x~x}|~}{`^n?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?If@Hfww=wwwxz|~msntouououpvpvpvqwqwqwrwrxrxsxsysytytztzuzu{u{v{v|v|w|w}w}x}x}x~y~y~yzz{p^uotz_{wxBIf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GeAHfwwwswwwx{}rwpvpvpvqwqwqwrxrxrxsysysytztztzu{u{u{v{v|v|w|w}w}x}x~x~y~yyzzz{{{|nFQq?Ge@Ge@HfBIfCJgKPjifvPSj?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfW@Hfwwwwwwy{}pwrxrxrxsysysytztzuzu{u{u{v|v|v|w}w}w}x}x~x~y~yyzzz{{{|||}}}P^?Gd@Hf@Hf@Hf@Hf@Hf?Gf>GeTWnjeq?Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf*@HfwwwФwwwy|~{~sytztztzu{u{u{v{v|v|w|w}w}x}x~x~y~yyzzz{{{|||}}}~~~sBKj@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf>GecbsxxBIf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@If @Ifww1w꤉wwxz|u{v{v|v|w|w}w}x}x~x~y~yyzzz{{{|||}}}~~~f{?Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfCKg}KPi?Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hfr@HfwwPwwwx{}ץdjy}w}x~x~y~yyzzz{{{|||}}}~~~hx?Ge@Hf@Hf@Hf@Hf@Hf?Gf>Gf>Fe?Gf>Fegdt[[m>Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf.@Hfwwswwwx{}SIzyzz{{{|||}}}~~~mDJd?Hf@Hf@Hf>GfAIfSRcgalnkywr|x~lfq>Gf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@He@He~wwwwwy{}ŦJDנP{|}}}~~~ѧrߝKk\EKe@HfFKee[azZ͒Tϟp|qu@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfH@Hfwwwwwwy|~˦zFEC˜Y}ԥXµ̩|IJߞMT~WUٚRQQܢezyAIf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Ig @Igvv wwwwy|~ѦoDECAbǟ`:EŴ˫IIKLLMOPQQ^{DKg@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfE@HfAGfvvwǤwwxz|~֦fDEDAAlt=;;C̲ϫzHHJKLMNOPQQZ¥|GMh@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Ge@GewwwѤwwxz|~٦`CEDB?BtD:<<=?ҰvתjFGIJKLMNOPQRWĥ|GLh@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf(@Hfwwwդwwxz|~ۦ]DEDB@=D{դP8:<<>>?٭f·ίQDGHIJKLMNOPQRUƥqqPXs~{CJg@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfZ@Hf@Hfwww٤wwxz|~ۦ\DEDB@>;ߦDȢa89:<<>>??DFBDFGHIJKLMNOPQRUƥy|HQnAIgdn{qv@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@If@Ifwww֤wwxz|~ۦ]DEDB@><9ަEq98::<<>>?AABCEEGHIJKLMNOPQRUƥ}R[x@Hf?GeQZvsY[p?Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfwwwҤwwxz|~٦_CEDB@><:8ݧE>78::<<>>?AACCEEGHIJKLMNOPQRVť|]gAIg@Hf@HfCKhepvcnFNl?Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf%@Hg>F`wwwɤwwxz|~צeDEDB@><:95ݨDߨC588::<<>>?AACCEEGHIJKLMNOPQQYzgrEMk@Hf@Hf@Hf?GeNXtr~vgrIQn?Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@He/@Hfww wwwwy|~ҦmDEDB@><:974ߩAۨI4678::<<>>?AACCEEGHIJKLMNOPPTtzo{LUr?Ge@Hf@Hf@Hf@HfAIg_ivwkvLUq@Hf@Hf@Hf@Hf@Hf@Hf@Hf*AHh?Hdvvwwwwy|~̦xEEDB@><:9752=}بM35678::<<>>?AACCEEGHIJKLMNOOܝVzwsT^y@Hf@Hf@Hf@Hf@Hf@Hf@GfHPmlwwwnzQZv@Hf?GfDKgEKg>Gf@Hg@Hfsywwwwy{}ƦIEDB@><:975308vרN245678::<<>>?AACCEEGHIJKLMNNЙ[uu]gAIg@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GeR[wsvwr}Wa}TVkqpt*twwzwwwy{}QDDB@><:97531/3æk٩K1345678889:=??AACCEEGHIJKLLNeuvepDLj@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfAIg\fuvvt|}|ҥuwwwXwwwx{}ڥ_DDB@><:97531/-.˨^ߪB02345679UzȄkG=?AACCEEGHIJKK۝St{uwkvHQn?Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfCLicnvvvvwqzY`kgqww7wwwxz|ϦsEDB@><:97531/.+*ԫOʩg7/123456?Гd>AACCEEGHIIL•bvvwo{MVr?Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@GfGOlhrvvvvtq|it(.>YVc|wwwפwwwz|~ĦLCB@><:97531/.,*'ޭ?Ǫnë{Ϭd?./0123457ѓZ?ACCEEFGIΚXzzuvvq}R[w@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GeHQnitwvvvvso{sgrT\wwwwwwwy{~ۦ^BB@><:97531/.,*(()+*,./012344Vӡ@ACCDDIʙZw~uvvvsU_z@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GeIQnhsvvvvvuso{vju(\hcnvxw~wwwx{}̦yEB@><:97531/.,*()*+,-./012342{JACCHc|tuvvvtXb}AIg@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GeGPmepvvvvvvutr}kwQE]pXnwwHwwwxz|~ߦTA@><:97531/.,*()*+,-./012343˄N@CCHwsuvvvvtYc~AIg@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GeEMj_jtvvvvvvunys`lwwwԤwwwy|~ϦtB@><:97531/.,*()*+,-./012343mEACCDݡL{uvvvsYc~AJg@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfBJhW`|p|wvvvvq}ephswwwwwwy{}ާT?><:97531/.,*()*+,-./012345DĀ>ACCEE۟N~uvr}V_{AIg@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@HfMVrgsvvvq}gskwwwVwwwxz|~˧zC><:97531/.,*()*+,-./0123455d֥F@ACCEEFٞPlxQZv@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?GeDMjZdo{r~issT[ywwwԤwwwy{}קb=<:97531/.,*()*+,-./01234566YإƁE?AACCEEGGڞPpd@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf?Gd@HfPYuymyjvvswwwwxz|O;:97531/.,*()*+,-./012345677:EIB<>?AACCEEGHHۛK~j]AIf@Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@GeAKkI_H]T=Cbww?@AACCEEGHIHəaZqBNr@Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@GeCPtSw^aaww wwwwxz|Ψs?87531/.,*()*+,-./012345678::<<>>?AACCEEGHIHԡbmUET{@Ge@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@GeETzW`_[ \xwwOwwwxy{}Ѩk<6531/.,*()*+,-./012345678::<<>>?AACCEEGHIHҡgn_YGY@Ge@Hf@Hf@Hf@Hf@Hf?Gf?Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@Hf@GeFV~X``^OhXww wwwwxz|~Ҩj:431/.,*()*+,-./012345678::<<>?@AACCEEGGHޤ[q_`[J`@Hf@He@Hf@HfAIfNRjJOiCJg@Hf?Hf?Hf@HfDJgEKg@Hf@Hf@Hf@GeGZZ```]x`TvwwMwwwwy{}Ϩn<21/.,*()*+,-./012345678::<<>?@AACCEEFGۥ_¦^``]NkBLm@Gd?GeXYm}zxst{pr{oqqqwrtrMQh?GdAJjKd[````[O^ww wwwwxz|~ʨwB0/.,*()*+,-./012345678::<<>?@AACCEEI֥h_```_V~H]LTr|}{yxww~suMdU{^````_Z&[xww9w変wwwy{|~ħ۪Q1-,*()*+,-./012345678::<<>?@AACCDPΦwd`````_}{yxwww{_``````_V Vwwwwwwwy{}Щk<+)()*+,-./012345678::<<>>?AAAG٦aĦq_```a~|zxwwwwѧs#Y_`````^lZwwwwwwxz{}ç֫[5((*+,-./012345678::<<>>??DܦYʦ|_`_c~|{ywwwww\w`_`````^E_ZxwwDw褉wwwxz|~Ĩԫ];,*+-./012345678::;<<>Fڦ[ʦ{i^h~}{ywwwwww w^_^_``_]^utwxwwwwxz|~̩p٫R<1-./0134567789=EާTҧlŦp}{yxwwwwȤw!w^] _o_^ub]wwwwwwwyz|~ȨzѩhثVޫKD?=<>AFߩMۨWԨf̧wĦ}{yxwwww䤉wCwwYWZ2[\ww!wäwwwwy{|~§çç§}{zxwwwwwfrvvww5wդwwwwy{|~~}{zxwwwwwwwwxwGwउwwwwyz|}~}{yxwwwwww wx|wOw㤉wwwwyz|}~~|{yxwwwwwwwvswQw⤉wwwwxz{}~~}|zyxwwwwwwvulwJwڤwwwwxy{|}~~}{zywwwwwwvwwwqw;wˤwwwwxyz{}~~}|{yxwwwwwww wwvw(wwwwwwxy{|}~~}|{zyxwwwww⤉wbwwxwww契wwwwxxz{|}~~}|{zyxwwwwwwĤw?zxwwwRwѤwwwwwxyz{|}}~~}|{zyxwwwwwwwvzltwDw%wwwwwwwxyyz{|}}~~~}||{zyxwwwwwwwˤwUwwwwwOwwwwwwwxxyzz{||}}~~~~~~}}}|{{zyyxwwwwwww椉wwrxwwwwkwΤwwwwwwwxxyzz{{|||}}}}}}}}}|||{{zzyyxwwwwwwww줉ww6xxw}wwpwˤwwwwwwwwxxxyyyzzzzzzzzzzzzyyyxxwwwwwwwww褉ww?wwusww_wwwwwwwwwwwwxxxxxxxxxxxxxwwwwwwwwwwwwԤww3vws|qv w;wwƤwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwޤww\wvvurww???PNG  IHDR t%IDATxud[qw/nZҖww(D&%@p Adfvv;k^9y9u_J;N;N;N;N;N;N;N;N;N;N;N;AN;k~ mc|QN]J9~2mT^i9 JuzְU?ˀlP䘲E`-Bv}+ZkV)+ܖ3oY,w]q]X$ ovڧv7 ںdjۂnhU 5._ٺ ,tZΕ;-y}˗ﴸƽ/ c+W]NN=eemNBs`^|.+g|0^N֞96jd̞re,g+X̵mhZ}*G7nYպmҺsewі M!-~d.0zИ~0ffg[6TѦHV(Etj̥M]kO|̾MfJWCEQJ?ջć֬,fW'J,&naLT'dc'#9O;aMP˫^wϳ>_z6}uivj?ǿ*)7Lm4ii۲rPe~QT4g-wJp Yah!{xzݍ3.i1s^2ͩԞ?Dumc6(/R;R*k.xϗL{ƖU7a!SLᤝ!W16@>r :7GT{-S)oR;:E*괵5h[<;^[#Z{d/sW-C5Gg-5qƞ喋_d;]hqO_dh=~_BZyg[ɾ6zW X u5/x;S)5%g9iQ]<"9D{|Z\OώgNoթxN'Wjnv[8n kImԟaukmV]AKz{~έen4-/i<}y;ѮwO.^ߚn-*GsÊ:8%m6R5>PJ*5!s)׈Z([?*aK۪òZJ?ۮNZb浱-Plf6w~,Vm䠚ߴ{+o飪Yn}jqmOVUx |1S/hgVw(w ŸBVu%W?mMGn̳uFrmYs+Rn{󗺗 iEˏv)hW8̋w_SP:\ӽqXe{qs^X0 T玀j4U$EE 7ސm~̷8fk6f=AڊhoYV1u6^յrN=]jGU_jS2N>|_VTP_sj{dڌRG9Kq&`Tm f yAfN}}r kڑ"gYU.sΟ*43fѪPZmQ^ܰ\+M\sNjEcמ'><ƿ둵xv}"22v`yLUXz,r:Rˌ/5}[mZ[שS0u''$uÖ\’Ջ<``ܠ s,] TLZ%żVDEGq搳QM mVT{άY֩ tƉqZm췪UER3Wb+>BhY4Lߠ3o"|ߚF[r+({EĻ}xD(_RΪEL4tj\b:*B fkhMXwˀg+o^rI3/v?7#i+_LͲzcyI^?V|Uqjq][E,{X*:M^NsV@g[85KXfYhV$WiG.l鄵0Ѡcf/6ۘUpѠUXnK)YɵP}V@QgMM(h'#},jRQx$YV~1)?}b4RV0^1HQ,71 { PsPHTjΜY''n%}l퇃o<Wg]{ǽ_Lv?rAcJ]][ke14/y`O,V}e6}u5ߓ?ީF,ccC4d ȕɑ8\PC4ӊ8*VBEs f3,ylPUk3yZ@kMT/jQp 9)kk&:Vdwn ?Ak0Q/MQ?z뫦FM:b#VEr"nd [J fJ"^D}KJ=yLK|uq-O~[m-:cll;'iEECNm2iuuD4dkk#gX Qtcmob]]$ߞ^kUA(gK3=S;ck}&4er(`LE܍zS(ǚe6.V9-e`W=OYS03:`(eKmR.{x܄&2l!_rVkOyb&fQ;ZJNAhts>z8`mkͻu ;q-}ji=\}'|m 1NX5o[jw6+ms>>*V1râbO;Ztx?vgus,irE~_`Úd@n1WN$&&qr:Vk d]fAaFno+ނtsY[W%;ca tDU1ٓ,o!4YFG6UQ43ǨZA搓-grMlyмS=@XT-sfէ^2#Tԕ_&sT*3)1X[N t4_Σl]Հ<tO)qq9&җ_{˳U~XSL,'RN1|SXCܲ>uЍK-0"*o߮g *lM>~E~j+[㒜MGdwnw6?N@lJFY閶V3kdmxZ<~uoz3-nְmk'7`V{̩eA|߳6h%*@yRU -0IP{(JYV/րrI~-3GUCj@HnKUc,yύ+rAS2TQ-SN1؃8\Ah'+vi$܅_S~Oy<`$ḿ\_n\y/`b}ok9 l ]dvv0눶Z{ە~t7v/⿾|G;-%_=^N9*[W[ly,_sԩP_-ݶ?qRβ:uMfWCJ&t,d\JdL&2!`21#ҕtG`PedBfL&Y(7dJ>dlb/2/J6ļHa&ӚdZ>vY˵%KeԬpnVͲdb:e Y%5Y!By>#KFe-B,eiVvβĦq{̓Yc3Hz؊,ElIΨ?zh*rNXGDuz5M_-Gx!ϞAmnПQ-R3hev:<+mCME6&$}һFg\SXۘܓtVι<;ۈwMs*Æ&`e[Zd3vϥCU,hii=P upo3Xݚ(9Mw̱ruZK4m).5@d;kU[n@j:VW4hV{{'i4P.|-hjrɨ1X=}YEmeF`)?6üNaf[{F|YΤplrm9rSv2xYi)x'1~/S6*[x:Mˈ2 /yֳU_=P: :bNeicY~Q~|~ ݂tWviNѶe룕s(?G4zA}lo6[ 6N<[c;|`S='whBmF&C#!6W`f{4i5mrd0!ӼCtDž6)rTO',Wv&蓞Z9ZBrߛjhc;-F[%b`ŬEgz|{༿NM`m Q(}kPƹ57f^5H0w0qiD|198Lnuj8.ꈫˇA݆rC$\pGo <^pSt'-Ùn0pu1Eaa]Y wx=|nOa}[ [}X17,QFkk=4E_jlĝg2Qe}kTq]ع,r_u#o4 ? ]{N#di֟5gN~Zֽ=ck4m}xD[@VmXܒ9@6H 9H/Y;]M|H#8H7X_tO71ޝ6![,ǡs4clSq f̝lt1GݎelK3=?:W%qLP\u''8w­]3EdyLyhTmxm d|vu ך.cr1&}4BL#=d ec#A:{/)Ɇsd@Y[Tf/F{Υuf(=eS=qW^tÿ,Qo&vX| 23W5>nRy|Zzrg{Uˬu?^8jjhv8}\Y2.TӗMqP>X¶#TnwkK-5g/,lirW~}A 63n'\DpUߓkaV lH[5>&߫uc^&;o|a\jwȽs^)9aTn9Ǩϓ[bfoP"k'E,Zj+1R5w/j͏SUsjCW.gtZrO qq';^ԫ{m/^<`mhJ?㮽p㙶さ ۶k#Ys}k҆2076G<*ti6%MW3g6Ji(y9&O4Ng(ɳip%Y(2د ?l6!E'FmMHu"#Vϳar&>˭"n]w`Z9|ploLdL9ؑaZ\G>ϫXKog66yUE`[P;+sԩ J2Sk[.QU3Urf? o[eNSU~l9 Yj1g]'r:ii-a2WbjB7'1@ӡĝ3T`oc]x|k, epfeyr_g8:锇FUwy$OzXR94wQ0}{Փp>8cƓ׌6Ssy7 kz@Ҙ 6lm "Y/@g=!d+g8(}@v V~h{j9GbstH2V9?Ew\5Zm H3+ǡuG 4-r`̟=\;(8UR^͎U{m5rdn+ad ޠ4{E3.fIEQtdZ)YO7arQѺm}PQEe0-1ܗdnճZLAww`GAd<)('L5C3 h}nq`2>͈?n 0 )OB,?]m? ğmVZW_j_9m.}xi;Оw/y$eJE0T,2W;eqmQdme8_jn5SI~cȐ L'qv6Bӎ#9>C:7zv&y#3s3y%O Y-i2씹N\EJ44g0%fEn;n$M %/eKI&e%ۓ)2o5yR $7I[Vt8 M!Kdͫi'8#΢,qtVGd\{d,Y. KdL%KeLY.oEl#ݫsUX2,ݳ 2 f!~,eIuȲbkRm^Y&"WȲ.MG1H,h +$y,]A8v//qt4jX0g(Ag$zUroFE4$W![Ja5o MO4<:tfo;'Z&\_S镧 "3N:ٌ"]V>&fȲ'яt[G:'Hr %ˤK-גHgC&s'JVkn%5gӒ/ICs?Y1s9,jX>&fVƑ5HHO y,=3=2) CvuSNsN!wSItr +r)A:#=ғNDK\N^n'fSbp1N%;g@Z.=eW&bHd| djlEvHjX6Ts3d|x({[dQwv'?O3PoҪ_nĬK3y7|E:J|ii&MIRMh5 {mh}K_L9tO0-5Po5 ڗv@V9|Uafᤧҝla;-}^q#S9,I75>L9lk[G)CjPK5ܚ;b}颪PݍwUzk2G眞LWZ܉-}𶦜5wK?\O٧W0 V(PWnsUWymf߲aRG#o<}ן-pkr!G|;S;giW9vˋ0kt8kxvlS>ԧnbݽW; _Ղ ņv^嗌 ]4WB!H1ڤ[ wczk$.wdH.я'1*OLnSC8- -lƼa0YvťdP֕t{S}M\i3u.&'LEyt<{}Kcp^&<_g姎v!q2?jyձ<̯G ~y0\ت r!(*=P`yңZ-RQ)7FrR*7f ltْh0ovg:nKj~O=({ ƳU *PjͤZM2ڬO4)uW)~{}Z3mOЮ9s8 :dՂm;HWE.a]{fjb__&iN?ܙ^Ū%vO5fX A%r5ldbe+% t[ 'SNA_Lxd!3`2ֹ!5D2W+ٶL>[@;Xmcy0rSs ݙ81Nfb 'MrE~t<l nl+ ,P/;zܭs6 s+"ۙ s҃b&`{*,\ھ]·̦PUؐMSYʝtԣwe0<)>Th)m #Dn2lӟrv9k0lSS,k2lR.CHw[),hp3rn+jE=>4=%yJ-RuM`ʱ>)0ZQ0Cۻ`YI2oz,{`2(߭xzg?ܭ Q/ASˬ\QQk"[~CʊeN㏽E/ܹrG/+'UM_?r\h ccx~i>0aEh5Ղ%ZLˀ4 !\[3PfhimҐ{ry%E^d~KZUW`o#krwNU~{luarq^29|j%>UW-܃&yZvr0 (<+lc܅YU)g۫{ 7&ʵXe$6Iy<;bS-=Xts^Щ\5rh18}1n,OfJ50}js4aE_.7a9mz) B~CxGKrK郦YbZ:Kot-7G'^v//ywWtP*s;ܙeÙz? &6ƥhpwo;'hO;iѳ0EZ9!\;& iԵ f5Қ[ӟfew&g8y3I%=n$##HIʍx,=ryiu>'cr;i#ܔs>{Ѱ\E:S;4fz Kg󨬞DVIWW|ߒUr_!d#2,5r?N'i";g"ge%l!hv:;JG,Y'ĶDT^s1M2Od|wN"Y88̓l.i%;/IYEdcp&93,/ɇ'b;.d|?]Ȏ٣6}t!?}[[ y,|N~71sT,Kt*'K4d\&ݝ03ȼ=' }ǓY$!;[L|m\6aQ>ǝNJS>"sqdZ=2'IgtΕ% p>n$<3:}H!|^LV#Iy &M+א^5i7ei Sy'wpy.Y=s#N !/ 4\C:)an&ܜK}k#2&e0ִ.34#h[yY8oquz)gvnOr\<2uQuy~c+W_fꟑ~@hZ$RZ2<2vIk9CH;f;Usf4eۉ ͏/|dlE\/y%p;(Cs55F ͝5,L?rwNo]݅MI`g~GKpFBxnlD^Os94ec`)–;Hʶ+raN-晬INUH]c&:hu@/z hN6q\-ظZ)ˢ1d'PfSI( }Թ+>ʹTPWI?tWy ʋ.]JX1r J1\ 8}t3?۸ۖkzgN5L׳8*'-wS6vv_yf܆.#r-=e>[z{(;aS#J_\]Al@y\V$Ԛ@uPn @n#YŜ29G5a FFl 5Wv!PGʫr y fК˯P*=Z'?* ՝nX~gް[e3>o3po޽g?/ ȱ_]A bKlk~RD|Rw*ִfM'o "V3!}՚yWǑip2!~B&iՉ ܑa@7~1gd-s"_H:!|LZ}3B_9/G}/mȅtzmW[`=.F % o[ĽTߘm[jXCki%>pdz+d;k$}\N*d y4[&u70+y&?$8Қl>i3NvZwjNҢ.҅lcOG9Cf!f`KZR @Z_!*醉}9 o<cƔady0(~Ɨ0u+`{MɑbT tT.DW2Q~ki_ --R-3옣J_4'arTnŃ9v{~W)fr##ܑj绡1ZCs 6#(Ck?_ Rv+{5/y_V괪`1c\QwnvgI'_lmw̺/<G{ w䄜?CTuei,pd DizGܑs2%EkFgTN#KGes7Y4]s*]y7욭;)d|1dl Tz!+9.'=2%WlOgY03S5W~ѺQ#1u~9ܙ/rleL?%=oӋ|FNo;G:WuLڍOLv'L0._mA.3a-]~ɤ+;*Ww)7W8"5HBi{bY}1ԽcZۼy~~ͥoei{Fr\[N[mcW<}Yy::d@^1y d,ҟ\;_5zDjb5ӟ|tu7МfHcE}?fk JH5pRzyԙ?3&}gs vEǤ[:Zs|b izy zr@[ 9 :5~j1Lv5̖ɶX2w`-҆(6ƠKr'Ж8<ZRߜeJ[Vmܝ)[MWqò̼m3X<&[nfr݇U47;VL֚hI6*n]CsTX~C>@Azg0ʈևiz|r v,:]͖Q9pt9ul4ԯiFyYWa^Fv-RM߸d eP,:{r;XCn ͽr)n&1MlYN?N˔]M>5h5 GWM9c(lz(Sؠ܎/XWĄZ;(Z2˖lA|c9vҜ>J?W#P~&PP^_h6v1썷K>d[UXρfST OWḾS(}g4sZYrԕ?xwOşgak̍y^q_ؽ~ >5-@mf~մ Ƹ34.nE5#3 1nчJnc*=+a4?y13o؅4V}As/RKoI!3-s gdZt݌r%f|?<-ȸ&|ufd'5+[`Ixvƹ7vB2!|#w KdN+1ʦsk􆽘vnY3ʖu2ڥ`Ȇ7Bl)S J}AP+ZL8[5X L*Y]Kiv +m>:ܗ[It\ s2zƔ}kOBtur+xLy5TC썙nHd\nlro= ̴'bj(bze 2/\rH n]U~RJ7>79Ɍr5*>O_F1'{јZzkg9KeZQ9sG{gxJ!Q9RPcmU샷-5GjlX+d;?9쎈({ UOm򽲕N#[gpݸ_rB.̀/\r¥_?\{oɇל˙ˑu#޹Ӂ^{wN[sRKlntZrν,6騍_|iWNX2!w^grEG󛜹)MT_.ư<^MKJN !mI\k>4gy]! d+wfCL>.#S2_%OAOo~n%tSzꏎ/K|MUB:wY;r69Ϙb@dƨ\}XţFȅdN^Vdj^6 m:듽Hw'Oo2#H2M<%6UVCNK3ZYi}hm*Ҷ#2b'*۱d,_R#}r'VuB5xTȀjF{kc@~R ̋pEjd-4ֲ߽6,_PGkt,mQYg԰'qLHC_6omJӖk{G},н\:\?=I}1TF{[Bs]N\>?sHK.4j)h$u9:zZΓX̜2CySr[(8N7_.'*gUwg8poҌs~X2reg#(perxhz\շKe>9l6X˂K{U^|<$ZO%}QH+7Pܖi5]^]s56Δ-d'ʅewʚ1 3}Y}^J<=;7m]>g7uw]ѱ>'.~3_Gx\Iz6\Y?p!\q'6Uv7fm3z%mZ]2wMijOvS+ch0\߸SۈiOr`^pi+i0ִw%5v~~e駙aJ3m4i}U $ϧo p iIqSFd^K>UUL_r݃ݛmV̀H^ܝȽ)I K71]nƬ| tб|\c=a:dPie6{m7ب2:?|'t~}S%',3>LLyKrP_}s9Vsa`snxܭV턩ɕ94c2(CRڽ쁁c?jy&_k2nT>GڦZ5Y c_tx8Kgmzu;UV>d~׭\y7){S2*w٭M纫qryᥥ&ElAmN[wʄq'ZR~WQM1-ZbwۃtϠ%{__2VWmCfjTV}^ ɧLJ|g0ySc5#23W%{o՟|!weHOr5N?韛H}}tN\Y`Gh&4g0ݍ'|325?#Xuu_Ce]2+GL&CST̛V gNbzC6ćj>Zrr9LxI_8aSr9"4kfy<_˿AiVb)|DYέ)4?}͸:Ity!w`6 2+M7Yzt薹ȯ4Զ&ָߧWfKfcM=צ\;reW.@KXfSi-1΁)Mz(e?,k,KҜH-Ƭ V6wٿچ?Pj^X/ǎ|኷?7__^0K<}׮G&?k#iMeP_ ?S$ԩ E('tn,ǻ9s9۾Ϝ O٪W-~|iWj炞8Jm拶&c՝SjۿjWWzK1CS {,tu#JsS myE5>7u򺛜Edqs5Z8wT|AGʀwsB ՛ԥwC  1 $= %rGI>6k?Cɰ7,]Os/LÙڷi>z =ڬ,:-L.,4No9Qy9O)KyOr!RPseX";P׭A6*a˅7Prt]O4cC=E2PB~Y.C7M.)WOgrePr=1(4\.l]5>.7$ esܓZ&MKo/70ʫ9X3\ug" uwRA8##;/ɩކd#g]pSNf>dXn rWzV͹r>sFd8Z?83 'Zbts y4s"25=JITUpҟtIL:y:ϒFT12=OF 'K1-!wi|r{^C}Jd͕W݅;ܰbu[ ,+/t^YO;nYrSwOE}vʕdcʱ:73rw^./p V(W,ڕ5>fk܅PN:z8ϒ57)=2tLWCIܜI pǹ|[I1 CIr j9H`!7'`A URGi***eľyWRN]=9l7/|]G3c'u3hS<ʇ{X>W|/-P^iͲo>9&w2[iμQ[fjg]~?k`c~^?SleOmb+|9^TUԩoHOoɶ"CҏtM/Ȑܑf?:̪j}t HH(J L ! tw13LϜ9wk\T>ófk7gp\n'@bd AZyd:ȩ"$$N=TKA*x%3( '}鷱.Ey!m g7ltI8 '%Bꀸј( MV2U.J"YG] rYI qK rY QR:JICSɧ\OGGv_ֱ[>s旓.ZoW|m+ݱ1BLINTkK7e {d}qN z0v8!!pGEp_ \4@]tVE Gm҉RksDH`+( fRC 3¢kRxp iu.TkDk1]\,_^ K~jN@mɛ[j95VRafe!$ċ4J{ L}Z5)j7f@o]SQMzf!/sO57M] < ][KȳW Wu\g]O%q!L]b:g~&{6{CHu3u>Zaヤ7/?Vx+0#6}GפRp8cBJ3A" g \% dS䔬z*iN2=Y.-@=LF\&B:d1@X 3 D*-sAb$FlH*D"\FY++@:K4- |RY ''+d;Kl2$ٹJ@bY\eDrfJnAƀxK"_<AKK`ng"?{A)ɡ_2$Lzj< uZ+wFqſ Xf )q7q]q1u!`:t }Pt^Y0IW `(e,%A.kAzKF 2ƿ_]*YKXr.kXH6"=H|'s7,K@N@l,2d Ud z6&erWECR 3} 3OK^e q*wAܩt86ꂉxU/Z!6~S8p  f̪ X qL=W|CU=rDZ+!Nn-R&l@onOte -j 2@ΓR@1K_PuM$_ ]MCa_W@A'I;e0֖p;ӣ%(EfHUI;Fp6HO HdtP@Z(+=B8t@IwLz*@߬60MQ%;KkuoV7 DͪRGKG/L5<1ɡnk@WTV*6'?y|_ [ ^̮>G%d,% &q:HD\pٸŇ\"%7 $Y'$" ror2,] oHz6 ` IDd$JL$ht)@se' ^$IV~AG{WLnQ;}d3Pw~@ Qu'8 EZ*;Mu}^5xiwUOM (..-}t#MjwT񎔂K6`-M7Qs"U5jTPKdB$:@ nБ7$5Q>)@ Gj<9nѹTIxr:SCm?'V{-&6浔PYLE '?Q˥w>87g=yT=aJLG9l:c\ۛ{sar?Fӑ~ 겵\%{.8(E/>N~ W+ Bzn2M30H 3l0)MADrL rN|JAR$R$+WN+D LY#\>n|#AH6 $ AD4`7`q9νH$! u{8e9]A"%BQ8}nLF"Y2URr=c4YP. kj j!ā4⼺64BCʱtk R*$QgU!$PL;Q7.*)q`.{@ >?JE[hH` @W[J;KW=1_\}q%o@m ji u,qAWւiyN}ȧ0wR xIE8Щ@4L%d?Nl%NW" dK@VYO\:e4Vlgy&&k1 rFb.'d:`-N?]?%(TJܞtw/Ye fZj?"AZEjJޒQ`j|v=_ :/-@ 8 ռaHȺQ.xR+y NMj; nK>q;3FIc#@jj)[6p4,@eI*1hVHjRK/PETg]}&Y51p|r탆gUL,=ƙ̡|zGiS6gugΛt@M08(] ͛`إµ&Oe'<{Vk5{8Ho9/DK%'&[c U5Ta҉g 5 fhE;&N7(JQ l sjj4M4yeD#WBCzS<4u խXlbE7w.Z}=Pz.~i`ynui,Ȕ{K=z)Шߐ01rd Nz,1oe7Y9Ty& %˙ e٭ wqpNB\n9jEhl4__a5LFۿH0NJ;<&z3]W׳N)]sU2(m txO6Zӟd7q{SPm/G߹Mb0n!H2uc HvMq,PpCT#!5{̓.W\Rr7#Vz (b i"[*ZFW*hi1T=UATzXy X? }_R>@*W Rv=L[tz9UU)W1poV=#x|4,plljSv!jtro9$;M/,^9s7~K4]MK-g+Tj5~[~N<|yl}*Xnl忋4GyBWg{Qms[(i'Aȗ J|(¥6sbwzYBd)Hl` @ "$DqTVe+@gf>KK%8ODfgy``I|%d+Q,c)ŲD'ty&d-[e!Nfer]l3@|,s@*rlӹÕYr$`otkSkPU9ueϔOdlb&O|muϪncfu$]2?d=k7 >n T4xOL o m'S4=<(_>/+C^_=)jG)AH|۞źܭqa+w\p:$ lOc6-^-$YF٨/@H{}vJk_L9>-o(2.t۠&j,{#V1'HAG{.KwnTjG ӟ<zʹUMOW_^x`"$bFӋ =V h=<@ sTfyb<$x'7A|r!bI|edĝKd1\ e8W ś [,3@L.[Sh~Gf`%evY 5 CR"ks D: K<qm< 2M‰L,1k!Q8zklep^ >NfX,@rz>YZz@~665=2|9|~md;j-5\U y`C=_q'< ˵2z:U-mUN7/vu|maU*K&od84B [`{h6dN23PxKdB 3dJ庽Ėz5f +~dRtNs;;@7SnI/Hvq$]}/~ )+T}#D{`j+ r:|?sj8xę= &_NJuH )Y!:+&ě,yJVDI2 ޲hxy!@%- Ń X)@`dHD"r[X$u9)y8$2iZ$K% wDz Y.K$& B-@.K Lj) V"W@&YLpBpY4.vA;>1J}'y_c=jssaYy& =8d ˨YAt[NLz2 u`t&ȑiHvHlñw%n7ARS ҁ@iB~7 XC% P>ewT?tfBl)%oՁ /,A^ *Amv L%خ}4/Q:AF^SD_GS+AɽǓl4+Y2[.=Wx??cRɧV@7Pж-W) %HE9 #1-*@t@i 3- dŇJ1W'&fD%s3hxp8*^I,!AM_ 6rW<ę/$r 7%- Oi2 @Arb,d8ngZ9(@I ڻ^w|$K¥7@2Ģ@ ת+ g:[y 2`jXϩ!pgA@u $/ %d!qWz0rS 8u{nBkHi@,J9dhT>&a^O nH Z <-}Ej '@=zbFZ&m;T kqԼ?S0`~ٝ} LZkYA`CʀVMkK,Z F\ߖIG)Vځ|N*@PD,{AIW\Dd.$d rF hf3C e* T/i f'A.H+A,, ^x|4bHig?p:dC,>Y8Sb-pY/ˁxY#Qk)IvíRo*\,+ > IWAuk^]l8M])رL_?Rsʪ15A͓;þϮUxwM0;QMQ g'C-62sǚXq=&_wKhhh⦺k"[9Yw/~Ӎ nWnh<ʵFI#!wWޞ.VYcrK>dF$pJm NFcG] X B$k2Nr5@.+i r &7(Me N[W Z2$kxOeqrd=7 Bdl`* r_JwLD`,.K^o/,Be>79M7"ڴ2FeϐGvǚB "^T3ݾ7ҷWh >ʏ #ټ? k7e)z~odxHik@7 [zu4]՚%\9N)O5%擺-~I7A^v\j\q" m=;웒;J*KIFrN.LK P4M=A`eASW"fj\dFy$Bf!H#< IdF,b eAxRh,Jiv9seb@La1H: \4rA>rJ,; zQD*H Af",L1 d4,q '&SeHLs(*og!Twׯr.\q {9 q\m(hzR7.ߍ!x|}  x(KN|@-V O>iiPsg~osk@[Au;nL|KdH:i70$q/+87>Xոu%eͅgQ@)a%#2C/0`pDQ &R(Nm{尌)(12_N$8dAY-@ğ$1  N;g#$/A6/ 8H/9fyHleWA |r` idDRq08Xd2@lݦlkLvQk y7o+Ύ~<J.1l ,f?\ø- (%50QxasUN{qީ~PVq(^0lqشOO>k9D%b -PòIDAT0/S.Y@gzZ֠eOL:,(@-,P%6qT| KK!@ K$ANH@z ߉b 2sdxZdDJ<ӥōTG<:`<HT4p[e|(e "Y4Օʹӏ++8<^]px9B FBU!c.ĸڸǹ"8#_ڀq[`-vֱc\laкj%$HU4z4^ ''%!flqMàhHhG g|H^_V[I:9Cj_|BׄdOA "* \!jzL@'X dt>\GA4A O2Hm 'SdXL}Tg5AЗF 3DY)}@*#@Vʀ$I,;r 1DJ{o$VV$EIs< !d7r;ܾj:hk uk~I%;p@!<уLs{U9 OjZ sm2cI\Ͻ>'֛,XTqx|*c J[m8]r[ :I{b@I psz!WxDī0ǽ]|OXV,8hHsR^>W ,+~@<1@ޚ[`)9`k0L|I^=>Жz߰F4ywzٝy=`y[(eK&vd|%S@r CTeJZ '\հрabe,GF>4 2SRd8r:6|9@=B@rJy @y /KA8DP X!!.\h= ~$oZ6-:ӄzIƷE0O*q&cTRY0n1\[$a'ѣH1DɌpTēHPF"0Y"G,,Ks@lJi&R d4cjЀ%@6Tg4F})#He#)G0့̑OAFF$K<%@js N,A4 R]DK M 2$]0 2(C}X]  hӦ~گY\q* @PSNէ@~=hQ  {ukG-Qʥbb\X )[зNgtsiqrE۔v=CAs7Rܤh- L@Wh홽ML(@ǀNCEuzƩj$΀ߛ>!8ȞO{ as-6DE<=lz7/ϟ㹤ԟ_FHy푺֒zƒJOLG~Ldǎ-`&/AzB&rXb X#WK1,xMHO/]ky\+ 5% c.Abn 9|`! lK lf L-zJ ި/JsESxx.T#aR5`+ FfZW Y.uiEmpKxӭd9g;w?1ֲMG+  (]/?s]n@ڛMcSKhBj;,湪Ub2(,'^؀I9 YTā t_XX<4)j Lyb GyNԦ0pQd1S`Y/8_4Hei?6PlXʘAJACeC2 # YLJjYW^%ķ:A*PPg3׼dC&yl-Fm0ݐܠ:9Ϊ)-ND0n~9[`K[}S@a\]2]IsCύ{gߧJP%L-lI~ |fzOӓ23F%jeZ #f5n hK_t;dG) wnxG$2a~I׼TTy!)Et-6p?e5 V1@Zcuz6 z]PEBmֱÇ.!g#7z2L @DT" DxO0 $ԒA%Y@Oȇ^*qbf%+MCVXXŗ4fp_2[2&(.Tf@e,=_(DRAOppyiL1jaKwH%Dk߭St8%'_R@4PC]c$|>'IЇ Lg0 ",60_Ԑɀ3 u5Ae9u@TJ|e:B6Fr13 (z_0Z I=B ͘+-AzP $JDIu pgw Ra5t*u_|M 7NobYq^ LΏ n!{~5X;I,0S$뺑FJLWBB*9X.eZYeYeiJ/'ڪ:H$߾+$DH9J[B)JSZO5K m#`D#irHm2@G da p.xӘI#` 4%Aԑ}Q*5&r\'LuouŘe<~66aWJ%*WM'w3]ypP*;BH -, np8=95cjSd(`0AP dAnto&g'qT ,U^" @HLAHG75D lXIx/.Pm4<u2|ibkY0>_v}e++êoA#i4Wz 1c2Y 'P! 44ă-J>f #D3hD~ L =p%mԓ K HDM}Hs8Y? w@Jh@nU:Tѳ`M _Q!%CF 9wDR/Ho먤jB@!~\P+RVKQus)Bq~)]ﻏ̺,U)f$Efep;8U"o._0 J᠚ <XPMu Anr:|x-hHS>Og xY Ki&w<H#tsxsg6[S#ֺx^2V@@>+Myu5ٮ+T9m,en_.N-'h ~&Eq;5Ȗ;Gs{dza[kCl=7 1b5)|ƁL!4w΂vZ;i>zӈ@PF C4JR r&Y!M^ wӊBE 棝~n4in73 нӜg=|zҼH݆wf`pSe+'?wB#8%]) .|"i 82@ctQ1ԥ>@ \QH.V*ʒ0pDp0N !e 8UI# d%%Dx HcLKKDZ2 w 3e6I/`\6XR`g5Uw`n 0|_}7P;l@Ӗ~# g"7( mFÍk/CcMq E/cIt@6Z 1~pmq1["XGJsRM9U( :1BbGhXu,4<!I_hÓF^jkN&p6r5E}y)~`Oudf/sKhfW[\4 i017>v ~߼g|ʏ@*w::tR'\uk|MM[?qmޅw_ rʪ*5.AtÌ(a"2@`聭(@YBgH0@'|Ua*r#A2RN"AO]X!3U(XICe"ziԔ> I8   >be,-@nџU|J: lr ({( GM'B\9q?!x#!:33%??= giG&xv*5clXt?m5$UNn[ AWXFQxq|[1 x:0%1tN?pthXORy Z)GIHzZnTV0NӆpJOIMw?*FŪ 3!b◔@uUn}, ˧OffOGw63?)\8srp>+hΦe9Ƽ*Cq߮!t% ?`-vV&&8jɔ3ܒ ƴ/zsշ?{\Rv@ϟKxٕ,_"Ѳ΂vW' !4mu) Ѝ,@fG 6d;>@6B )% $ :S 5)dKYVRX(+"И<-I9%ԥ@2$ )j1.IO=EI d)Jy7RKcQn^S럊sIXޠm!=o~C˚~>tmM>i^6sӡ{9*'?7o|US-$5 7Ο/QXTwHb\51J)ں=fFCRRǫ˰>)/MM. ?yUq7`# T}E rjs3y_8T _sHIG}A.Ɵ) DRƁ!빜Wn߹S_e|#w-kO Ԛ^A`o,lX,g 5Pz9;'|r(}#&my9Uw>rH}N-quǹ=Xޔ44Ξ f5$sٕ?-0Vߕ8r\|GL.c+>D\hL &34.]d XB T @CL%3{|((#-Dr<pHLU| OԖ9$-LR'9Y+ gE^@O Ix柈 ^˞VݾhZդqKOgDNPKp"EkP_+# <9ԚnY"бC[.\Ks"A#΄"*v# < 6 RH6 5@#\z4 >/X RTyd/8@= #\N4$$9%5rБW@n0HR!B_Mu 瑫_ .aV@ Gp eVw>> jVT6ZJ_W_phBc5r^;b4S"l J8!䨤 N'.e; RFOeeaL'?Ȟ^TÊWH7ķ]&slqA`.`JT&3F`X(n쭵pzSlؾvp kYHuiHb'ĿQ$, ^}e>'.Xe O g{'Oٙ<Ai—R"A1<'x~|z+s!kjckH'$`%y> 0[LY6qqq%pU 7p0[[痬6Z%H]ZgiM/Je~Ņ=H>hmohkaݎ=%TOln;Ky@, c`e(UK`31Plu8 3/`"iJCҁCeEyJSebCR(|yFS lО/*BC ,pUx( $1A@4]b`[EY}(P\ 8M; Pomڼ+M `LJq_Qu;,uFA(˸Ғz/B(_pgzC&7!cۧ{W 0Lxl%Qmq䁔b:)=6 7̃"g@XIka1|.!nOU1{mZ!y"̠KaNbU@wuq˞ ̤ DU`GPS˲{ ^qu&潃oߗw&1Xs,Bst37I`- Or=q Pmu(@A ׉HvfS 5Y'ө 704N> \2OlT&3AVb` 4“ȗtg{hڗ@%hUe?R9Ƀ!ǀ4c+@(2d,X.Cc)"cPA2 -`&Vl@ؤ%67BsL2 0HW*;?ft( ! -@&{qNe}]Nv<30*leMug>4̙zIđ#IʇMPO|HVG*]m3XUMVm@QxG} 50HW頒詪j Uj =Q5|(іx; ,qnd@NvPto0\hCyuU ҆\`t:UA׏U+k)!#T q_mY:- wܖ +sրu#ӰMZ&9(K8Ad]}}rz'r\^UR' C6 I ޅ|@ ~ȳSyZ:g!mn<[d_lS!:a<;s/+ckl8%8[d G+DžʕB*z-fn+'~Ԣ&xF  G:Pִi:RAc>Pj*pD (X@mt"JG86qб pBPUC!H; &5!/D!j/RA#դ{d, T0ydQ;BO̷VD5:4;eeࣶ &<-Ur U K% SV@&@LAT@S2 3`THU: ވ:T LPu`֫/@g,6Es#c.dPڥ0D(i u+ϥ)5l@0q\VӔwgAjplPt >O rLU )9!Nye46vl}lVBJ)o̒_l(c! \ U ,rIR9 L@UjS m0H).t T"4{k%R#'?WGtRŇPF(F7JԠUaʐ\/5)8pCStd.%3Pڔ,x}Ų!FsuIC^&ȃ;rђJ`0؁ ԕw.h dDEg3$q{`M/S^(N? ACSEw4u"(u}Po׊v$}hUn*ȯnHPhNcV1I]Tu@TTuEuBjLPI*1zT rWgT$ʤ6x򁪩0!*3'Tj}* T-g~-?Pϭ @e#HfǛ^`"xjv 2Y68Lkk+=ezCrA%|Y{N@<9m|{#)(s[szewH)Ϊ#ckts.0c 򡌖y`Rk*Ri0C|(M*S%7" pG.9'cST@Ҁ> X ГJ]APЎt XAC| ̀Oz1xČ0C|bzl%6zWz稫I+ `V<K|&Is2kPM46UuTI& PU =@S' ZU>ʥF1JT:hPwzeUQP@UZPzrxuSUv+Ϋj h*/8Y`eXnnQ镯V*5*P>yD2ALmht {pK@*}];%CP@Htp(,*2̇S0ܲ $wؽ `2ǿ;h)ySJ^â[a tÃFC}泄3MwM7DZ>)ӗt@c62J2r9:bEowH!M =pL%"> ޕIV8R)HOO%+SY*\*<Nqc@Ko<@" \"Q] h@BуMWRn* 64)J/H1|SIaE( MpK_u-O;@Ūj V@=jG/PM> ʫo x[uTFN^Zo LeR;A5S-j*зUsz:SnPoj+ƪ:5TP9/uq˭J.%),5.4&PD߄'}O~.<8]>zC9Xc^usS {&3nec&1Obn&o ʡ")nʣ,ZÓ}z'@ZOk?g9di6~Pn2 C685A;"2J)NFf8vQ hCf2UmQ??@ozG,PiT Osk*huSᰏUo!c@ F}5/524b4=-K%N.`}iОs;%pK7]u5w9^9S} }I 3-?/e(>䃀 g9КꨒZyKy~5a |岱Z~2zWas /$^:,Bh <)C) O6(gaPE> .F7T /GDwCбe?_lRH:@ކ+ Y޳<lV TuUR= `u^U@SPyVQu)M0Mm3fj@=WTUk@e] TJSfj?qjƫg2Un**T:P*\Uwc`ʣ1LQ@eSٕInAZPfvz ^g`lj2? LyH8I:.c!^Íp#3# 4TT[ i=kAhͰR| 5SQ{vLIiEh`52nr۝|$c/'IGy\oiRSO4?m?2|*;$KsP-|xZ~f:cTJV['_/CArT_̤_` tVHi pjKRqG*!?l&M\gטYvnGuy/&Ҏ\Fi|! @S{=xjVN PA}A^So~ir^L JUT:Qo*TUUDUTuV=o! UԛDmf0*jUajR*Me||Q QQPWT5T->U@UPT)P%azuʩoA= *zj jC&sgm<(==.(~E4`K1~L4W)8HƩE*&]Spg2!CTUNDM"g dZt&oG?;*Λ9TQ@AQQ9bYۜ1v2ts99S?;ӷ{>:kVշ 1{_WzGMǻչ-:9xы^J.cECM֍/T>䋏iz19D+m+0QVPj4w 4X6s!n-GV wώ͞w#=g ,wyH\+FSٲ4 *\Q]U6M)8bَTbTڊۆ!lj/UA\fQMZGte*P\R n%SvW iu\iKqn5^?QY([&~Z(+Z/j}=Yqɘ/m.[qOaJn Q SL̻&*jcwKAӈUmchTq -b}y>@T*ؘ;;@[QlUTeF܄bmHL4ZNgsQRxaqdމ (.IL *}?nʕx癪{Gb b\? $&x 'Sbf+CÎ)oTk6ނ!V6]<%_f1Ɲnv/T7hJdmsBnI#| :vi_Ĝ~8k4}cegy S55qe{W\G M_k6vS?|wO+,g,w @6z*E^ܻ|w}nhTcFA\S4{ֳX{ٴX,<Vޱ{)j$.5X~SwQ.UikҨ zXL_;P_IZ{6AoMC5kT ݈iB(q>Yw_Jj$[@Sj{׸ؗ_㬆As%oTG\\JsvDQ"Ig#:C$g3=ms|-+r|aFF^v"pa>󠼏]bsr|I~֑X'o$vFނ_i6d#&-S\+#O:GcF>;ّ2Kg߹nJ10 sMmn%4C>S٫ʍ޼w=^-~Sy~Rj6/k't\&f7Enhg=kPxןّ~?pכnmz~::Ukm4tgC{RB!Ɵ|{mc8= n [4뷸82zA۸T+ QmU>v+SBCh{V_4_K0TnVN +]15h2ZkUChJJJ~oR"yα-ܝӿn/&26t|2WQ '4;dcUkȚ8?.}qA9ץ|Uy;9,?J&ω1vY3/sH^C̙/Pr|ùd쐽p[8y=NˡYo |vJY_-옏@?ϻpmgG9Y:.s19:dOdKGg<[q<7t=P}p `39M{h$vsѨ&ۗtΐJg=]7\5'Ao#Zt]EZj:j,k{CU44uKWmTjٺ[UnY?iM.Bo":.-Z훽(.P\q0T✊*~O_t;K-+W|z֥.ct]؉l?h=?Kg%ɉ(r\7u0n(bvܟ%_~}Uy9'&W`}Uc^zДL G9Q0E|^V^\ȏBoW[2r#lEQ;(;m(sߍhkEz{諮^}җݳM/ؔϜ_aNmZ8Xoӈ5paJy;Z{Lb"j08HY?"1BvΨ;h{Qy71U6+o#T{phkdފ*!;b q}%qAޏAkS5cjWngzF=oMm⢺'f1>D˫s0ߵ/[f[:G o.|\;* 6)&w;g9lZ.!뜘wkgU>Fr9/;~_8j)rNyv#;.xI~sbC֮}zWWRQc| b`^?,?YA'rE暄*U G7;UKb. +=m]*~_9lkrGٴqPޙkUeć>bI.{ih ;?LΕVl@~Y2xe,YaUO_Q,?=o'W:9_ԦMV>KJ08k1ODRtD5~2kGh_Yܣ(n3SC>`M_+渼 ǗO>]3z]DGNTcQA SM–j>*).{@q"hY@w=c3>(GϸXˉvƖR6[#F DD!NťzjOlmdދTA* %6Dq (6T%qHN 88OKg||Sp.|;DZP+UGÈu)lg{Xm'8.A\?6i_ͧM8܈nSM*w~|}GR~ՃiJC}So?oC~=ܘ{T̨ڭӾK꠪,~fݡ %?z:9k[\f%*?jݳ`s -`.a% u2|#m/Zcu7 l zf ۛհy;7/SP^ٜqkdJ|[*e^pBS/~ p1N}+;S 1Cs#':U9  :ݝ&ٝmI#A|8%OϏmXr#]G$Χ6r#_'_7 Y87.? y +oc׬;c]s|e4?ɮm1ِG}){3o&O|V%GgN'8vv)-de~≖XHˇ<Ц'Ƥ[QwqGjmr\܆B귛]H!ͷh=Ȃ=JzU[5E ʆ\}Rw6pM-a[ mq='C)cXcY;BwYNOߟ7*qrU]jPn9?kbTPX[iyg݄B~sfy?w W`V rf\\)`GNլe-tS.!4wԢA7T^#66*A6*/T64\˄83czF̻h/YÉ=6N .'{/V.ZX+v&9s#=|y|cF`t 5ov"~f.{ >7lڥga}(4Ǝ>FlM4/uɽ!VX~Y}[ _tc4n g|G?So<#q5WzoVgy!kRU45Q1Fivcnۦ˪K/_]}^|[B.qolOWR?F}G_JJlQ@"x-o^l뺿A1)mXڳhꏪghy)m@-O6o۾i[ٹP˽Q벫m^>,̡bH-".عszy?Rv:7vwP1;Poݻ;߅슸<ϳr6ƞ57Xur+xsYDogwRQ>ӫ˦DQ||e&ڲט砖XV?Q -!S)W G\xr!1->uɚ{~toUqE2ٱxJ1?:Dpdsc 6 f}j6]N~>fcBu#l$[[e8ľ*qb/GV7g68訓ohbYkt ` jM#ÛdKWy,U\z>--F:Ex.eQ?,͔Z\M+Px4S)Szˬ2i@?9׮9<`ɕ=Oku"KkN }hwZ7bpF[7KN)4\."Zإ%-i0g4vLGi~`eͭYwj ڂ)\O8YpȊE}y I̠5~F|h::>iO7ql_ޏ/_'4 w'V^]m-CX#oc_e)(n*v_+gd<>quq/&Yr!.Ug,ρ֊itSY~¼ᯌKE>RNϱ41calCc"1+!Z8#6SfmuԒ|#7S -QX?PFلq5E6t'º2ufoTzӂn~ކa-DX cR3y;jtp3l5=Zw ~-ȧ4ct+֊sATcyVV1Zk][+1Cm.+7x[_ȇOX@ƆxD܄պL3NmӍۢj­\;o"c n'}< ɹ!=v4ȷ>jGV\-++/ wx~k' 6QmY"8+ ч׭*M+va!NGAmɲ,꿮3(*jL7[i!m.xкsㅊ[qKʳbɲpE,K؛7GU1qg\Wq+qc<O{hPT^|,ͅz=q~{V}}︍]q˻|,@ݏӟ.Q.u||r쳲 .7βk/K.p|<ݚP.σQ*S-j1'Hr^˅d|(oƙM>3KD1ro6f93p팳ȩٌʙ^[Y9?'?sA)94/R'ΏF.͗U6ɕT3dL5A ]8+·m@6ݞHt9eUs7qn/pF)uW!Avʼ2gpN~,mTnBwN֓)o#G(Yr^~k;yǽL]UXxȽ P];THv^FՁqO7||ɬmݶ}>bƪ翋i9=/ݝwܧ/Cq\=[JlIւ;?Q73u͛c}J`9O0%OƿC qPIEoG џܣlTlTg"rE.g6c"r!vBxIuѤ2 w_bƷIhR?yS<G|JBGjѠ{_7&\aM1ZF0њJDe;Tc񚊸ϕW?/? dž3&~Yt-yI=+Xue`cIGl4R"y`J_lceG>W B Tz+MnFU0qu>Qy5c< \^;;]Fvے;`Jc1Ab]Lqơ: }_brTL w݉V5'XL!@ &NЛhoL LXS %8c_&q<1Duid=yiuL]b?ݱvZ5yޯcKbXHGbRb~~Ol^OVZ!JAXc]m`)9_ kAa8l9Ƽ^E%nP_9$*v.ْt` k(W5Vle R__x)~G Gհ-|c[,~ZKChr+c]LəZlfym?zf!"7.#;M1<km2}7fߚ=;wr]3uUm#\.ܦHbӦשj㩦m#,UUmzbTU~~,oD(ȗX0$'(x? ronU1ܾ}+}osT^E^G#܎ }UVN˜ @GUb s4;'xI pw z+6͢s~O?F\R rW'fN{KƧ?EqӜborH> MBGKqS‰atM/Mo<[ssyD֡c;`F}#|/=f؂|?& 1?\B|mȵrlkYG'2)"<[N#{g܀\%W\%ʧSmk;0r^^Gd|%_r) "'9թ)9<\ycHyrìjfP|+7u}U>yKLnO3d:!WqlBmrŷgE y -ha=pces[S*{xk ;MU3a5-N5|};E#l]yBN"w-^b\琇gŻ!/嚻!)(rPȅ[nţI/^I+4,[i䝵4u:ڙM]\͵w<o b=RM[۝Ӵ2M}pChVw}*>yy<܏e6RoU=VzD7qLu( hcePܵqhSƿM7#JjUSiq<-N.-l?].+`Rع(S?4o}YTZ ˥O<,krؿ  0%aDPr_lU`J! 8Voqb9E^@EE.s'-*iK(\K;(XcQ"mTj 8zqm߬o4=/iIsQ<og! 0&sΙ 9$Ɂ`{:osˇKsϐsrV_9AYkr|6gMQ>CS0f993;gw6yU^O6' g:8#G(m~_򆼑\'_Y5dsG8'ծ8=qbޛ|Ȧl"ȥNq*;kk/ 8%˛Ⱥ-)fΈxCëWao"W_]6(,-;j·NT sj=.{lV9 u (SGke[abəy:qՋCՌ@bU.~<؉V\y}eKmn7(}\Њ-o1 Ee"6Ք_Q߿{R8ppĠ fRqI^B]5=ذ3y~ix!+\F#./F5Rusq^kM]A*X}R~^B*W*GDydko5-s/,jw8!ߧBܺnŏV?T%VU]=klSqDq ǝME5_weؚxb|ê+FP&m!,lOJ>]ƝEy9#^g1O, BSb&R%U)QZX{8|bC[[rFi,Kg% ;Օ:Gbxܟ1LU]Frjޣb{l"chaFmM6pBc3F5@ >ZmkO1Fp̆lMvV^ۇH˿+d6e#KaChF&uq@KlmFZzh1d1`htp.I[ekUcsmp,vg~5pGSGz,?,A-V7r"M s=i02s.kQt簏{s1ϥYC 5WGb^msw qfn;ѳC+ro 7tȩCo_5ambނ(wU3]uWO:f>48Y8:ŗ#k{b^4<=iڴfhV8<:RhaHU7)T/(Z]lG݈J[u#"zbI tt̲ Kց ySQ 4O3ÜrmJ)sHӟ7\43J-#L+εVxNC4|AĒ-`fҽ 6/?F'XKQ ܯ"qָYTij~jKʜyy^-c^&2Jdz"gKQs!c1)-ƣ tϫР1y&rR`?ҬqlaTCR\bSI~yqW~ Gɂ=u"5<_D/įs>)k1&;x5haLl%/NiqfZ:v;Qig08܆~p#=|FkzĎ =R3OuۈU\ӑV;Zy -;P'iNhdc$+45nL -#*lOqo¸?oGXWs`8-u7|RJb| PJTqXn5ѢLSxJK͈4]q. ֵձ<|$'/=|CAo,^ ñ [<7a<_{,/%wkv!&/.U*k=4G<4ML5 ?68I|y-G-r=MFDY4Q3mno\ƵY2d:_o}*Q/ODZoJJstm~17,o;ىa%' ߥ桳ryMUKh]zvUΧžk_m&8΍'h4ޫA#93μJ5 GʼMO{N Y (hV~7Z B?`JbGGQ_bX{R%M׏Xrڹ+~`XXVU>R99>WucV~Sh yx倊)kx̌ce%"ץ׺jm+6_Ƕ[-wX} օX1*gD 3^&V7+b` q91o5q-B' ~w juA1$^&V򻸀>nDSg0܊q q}+1؆`M\7Ī z}bTgf4Ab^M+QK\\9vUjew5MbJtbj܎V_Xx6gKQ9/ۃFe<8KE8ݛ_Kt"xqNA8ߗmMUY>FN'5>#qw0K h0q4VN61h{ P -r9H_4՘m8 kzER3ʿ -C53zCwh@k9XCײIwX 3}ǩVF7 RKܸH$Vq3ں^2w[l[6\K1ZJb\K'zd |9Y#/60#D?,`s|K[գ/o߀(o;癧Ewx=+:V?Jm0#:s;=[B՚ѕT0/U[kUNX֪Ă}+ ΨW=B/,L>sywn*y0Frx>jw򋜖ǓS9/(ޛ+kyNj75 {u'qn<,$~ ,g]j]蜿hlr{˻s8 `br:V w#E{>eU_uc#"At!*{;!4$bonpRz*e+?µi<71X\gx~&rTV5+O[H"wɏ{)Llxa! NS=Vit<߃VMu)>+/AV2>7C-6G 1$qi ]hco=ckoK ׎ݾRkC>`SlՉqBnP#(69C8G6PbHl9ha}{Qi&k)=D>(qxI4eWF.2{VTӎq,f6@E>X [SaS[ש?j511IQNl|L¾ֳV˩b0=lǑ{[`,rOφ bq0Ӵ]_M5dn~k勴zLM{C.+}pg~^!7B)"—7Wj7b|=<.BU|$'P]rx1We{477)Uv! *ɉ?kXYI{~8{<]=^ށ8ii}t!-@C56G\lHLt#nk_͜}gNWؿyTVuL$ 3 +obLՉ]7+ϯށ+wL{mwɐ8ʋPp{֠G| 5ڢޕ;{ظQTEΊs}v4mPwq`\z>X19mY+]cF;}}ww'~doĔ<'r_x^G42'e3%\LR>,Augo0$OɫbGXDlBbv򞘏1ooj$zFɑ2PQ=>D.☢>mɗA?sC}%1ӘFna {TIj^.wX4*.ɑ +09׽|0[|M2yUXWi6q_lߪ!9xLȼTs|ĭo#‡9i{z7{O9l yTJ*c8:k3rcl0mdlRF51se$:l 3l5`5cG41/ts}a2gmc10ߡk,݈\]J628X8E}'b6s͎a}_P`#x^Gk!W̘&s F%4aG+kE;A6:n;C( ~NmUh{ȶhn6ͼXח꾟IץU+ Cay-yytQ vjOx. +GA%;`-6U,ss'{#d%19D\cc|$;GUKOq˸d*74_8O4.p4cd*] G8<~1b:4<ʡmu+-8)|{7<51\eMFUN1*Jܟ;{{L6bk& '.hv:_n+W[',_W+af~S^?7cv$"~27ǬsWw(J=p(d7-̧[ g-1!TMڸa|^-ҵ%}q5bV>b)6k܌|\jqi|z\M6 >ϓܝq+ykԢʖLYX,27 EmNWX6U3.9ᙆ`Ӊqv13>Ȍ}iY#0ԓvEza .˅ mL1Wt0fC Y5ɖLśZ{Q6ۘi V&Smcgf,b=fTe&f4M*6ozo^wDsD-kVi{IPWk̯)k4eMaH{iy#.b58f{ܔpyp.&A~?8m֮}x_}7_; tlN^2\(WQݹN6x_v:]dܲ8{՘=C|Y֐ɉ*[*+ 3EO婘8ոB<<>At3cXɈ=(8Rz 5Mc-Q~ĞyP>Iqk*U*qKArR)LO:r VN6\*[x!CY) 4x ;U:AmOX(1}Ci-ğ]NqZ%s~Hcw`1q/g*GVF"fx|輏c,{ 5Zc?B-c{lwjt8(B3tL(H5aCm`mJ.|XUۀ|Z_gŌ ]V&cwl}ۣgޫ=*q&Dһ7̗ahj*w泪77R"/۫JN(NMCgy Qc.e-nr׉qnNNlۜFƽK(씗5\/_⸜G3%ijǍDkWĐ/QZҟ!<FCWn1>ޡ0,=ch Ϝuu3>1tsO 0į* S7X3=lg[7+ZJ^O4aEkqXAXFISt't]Qc%}7WJ[:i1Y8k7ǯB;-DO]M" ? b83q[QmR`B}Pk{MıqzlhJ,ODW q<[Qo vI!sʪ֛VAKWhbhY]Zz4{݌+]vVMha'~Xk>XjEmіgǙ{|=kP)v[V?Χx8.:d?s.qjRӹ F),xx],`W' <<:g[X[`\U;gă9έ8*ܕ⺮~y19y b:ܵߣ ^}o!ccOMś]~B3z]6wgSqnSZ%C]j(L.ޖOoB|frrݳ#~~_0ڿp gܖ=?!Dluph^NTeiٱ'ϟ r ˫/>Y*WW+ +hV MSh*}q$11f;ķq9c Ilks61N CĬ”xU 1)ų6/"]UZI0JbRĖZNU\Dt^+ 9IEbLlbB<7_1XZ?]RKqG;x UXeX,cLA*uXV+jlh:&:*EX]k0 YM&i(T-F*Fc6+*TÔ2ZuI`tlqz9{rG)ꠝF` }Jax-RqzP#,;h`( B8Z-quBkQGcVFFc+~*lNdQe%3)F/M18LdRr8cv6R*{j(*h@m[ʬbsT+ .词Jԛ$i%+u}i{mi1gә NvM~HܣuCޟۙR#35s1L_3<ΣӘhT1"yQLr\@ڇ>ӈ s!o$ݛ8 sO^o/.{w +7=Z%V^qr͒=Rqy-~# NQ7"O:y'7ٙ|( ŧrVcdޜ3$;P\m93Ȼ'78چ#݋فs"I/<4E;&b~f.޷{syw?ONXc.;Rʬ]8W>NZ5U J\x,GaX 6Gl#cil7ŗDTN&X@aXōQ$&N GXD̍kbnM?NE\bňx0/*{׉-x&1;˘Du<|gRش01f"s;S?代>8hUcJ8ٶ`SV)W.823ʂ?MN"bM2GSJMLWfkE-q$2ֳ!d'mG*L@5v6";N('>z#jF8Θnm*L`:DŪ. M/Y[*ccJU!([j-3s*DzkE`sXJ܊]Iլdڏԡ^;mMT֌*Y^!6=4))E([@ PLGc mXUrb * q]ڵU zmZI1/#pPd;o!;j+TO#r.r"dC{_AnEɟH1jF<>Og%rh%3E,i-=p0=0ߛgz0"Xv&t:f#7eC]gGb/HNٝ򩜕S?w-yCI9k&ySwsb1vۧiNӛE 7^)^ɹ_U$9љ80p?l\繦a9ٓ'T CE~m,\s|F;Ug5HCqzSyW}ԉ,2f磔?`o}L^?WWV~WۉΫz}XPא̧(>Q|*!ͥهt'\Lor7-{wNx{~oK0eO,;szEb8߁Tм;3720imҴQڭl{oz?Yضtr!ѻ.z|sm-OU!"ƕĨxmDx:" =ou⎸”zbBd|@tsMO|QX;@̍H"V(lZnR*zg˅U7:f/nJ["b((ǽ&~@tM]A4k>n&FbBl/m\[>&;}xFZMZ t_YS)E{ *_bi8%GGm4)G͌#lvbgѠ tP.zR4T)LjTo'iG좽XZs(S b1m:^w 1ڔw֋LB!fh[*V%.TmzC3{ҾLlVٶkg`A@1gŜ0b$&9 92'@2#f̡9{{}9ukﱪ5m c9|3F$rNX:>҈έ$l ,'HZG'⚸Ud-tXVYSW4*d94;Щz:c GG}]l@P4iQNb X%Pt^QR}\ՖgeήބmvþJVcr< `JĜ=qUsY\E 1DZ'?9Us}*]s'Qtubx.1)f[!*@\@̋#wZLoY\Kl;0ގYFOR)2f cˈX'^% GE{ϝ{[ciᅬhꃎxC[Z5f#cW4X`M'#bdcH]r֋1Žv!6 Ӣ޴1RFl)KPj8qcse5I&Th2nt4 |E|蓖aO (L-PZAuރ6֫@zj?OwSKn 2$ks0`si)+b'kL&C(>ò@>O]nA&\E@ݱs~ޛj_;˱ E 3UXU߾unyi=ktðRΒuT$9;MJ [Yf,gk2=⻸.\lFCHxĠl6c\qHtsLj6|k[|GU5 @e`iakwim4kAG;Ώf6CMlhnɖ+Gi` ʅFԶZZޱ u *k 魕 ڤBL`w5]Ti/,96.۪"פ(Pf&DTa=lGCybml*hA¶U%֍Z ׀V2^Qe+1`BB])`hD_}D]lFeƢ- [Mj[oGCSt8Ѭi`~[2p->]z6ԑcp+ʲ{^()@ +;yN͞Nȓ}302*Y{JyL]LbydY}g*ksY4~waygPe]EzRxd-jz>.({m#k+mN;iA<%wSeV=Ѯ-evp6{|To\ .]1ſ6?@Nʵ/ٔ'7)#嶹%lb!ۑo垹+٘/fh097?曙ȟr_C^bOFKXƧo>]vz[^ JN#[;F 8+eǷ^0mApt4_5Mo-x˵Y;m_P6{1ЬU"'^ǵN#-\Bt /&_(Eb*̨xZJ+D;b\J ~9<:3B&0{^bݨuQ[ x2&kQBl]Hw}ī^Ntq;cLj5n'VU s(lQh+*Σ9'܍ڟ~{b'ťV6[H-s -ŖU<1XъeFWҖq8ֶ?%j],8Q/Pc:kѷSWTUmu"m}㒸67Vf֨5zۙ823c}UX2g4@\hݱvVXRJ,.UZ8 j %qOq@#uW2,C\/nĕh, j3PeX5m3q5P#Tf*V=4.zYXJ;G@EU(kyr:Òu}|lSrt~gM]r#{emLt#9>70Jpn\B\h|9 ,CPGA"ax5/v-ނgr 7Hb֢ƺXXhhK#zz͚PH£VYsLDت.v*+L>-B> ŞG|u+I܅ g9z>4&[効2CsvnK(3YȺ\;Ȟ9`׌Y5?'a=f~Q[8ZKE^ PLΏqgl ]_<ӎ|_;Ҟ7ϯT~v>@e./Fd;U-u4万X>6WBqīOգAa;7tLq1]L2b`K+g7'/GX?y 1E%*3 Zb($66ɵDEtYqO|BF5αqL,UֈB6Mf$-tn%z)1!Յnsir6<;aqWR[_U_lZ X1Bm>D bu܂ BYk%ʑĎC+TҚ`oG;nvCEG}wieԪ'Nb<'c-s_0YCVѹ!,oМZ=0˰btt˥؂8x0vsËy;'+1& t_xGG2I.;G[%Gٔ5.웷(1/o\Wܘ|6, ''s~nF8{9⓹(&vePANms\)ۓk3%ٟ|7˽3C<(˱CV ='.|8y3?->T  xl_; /yuL|XuŻXv~:~5^ mI ǂJ 81#6rQb Q 1+G1(G1"1:8X?F[Žh"> 6i2~6eKOclcOqoCb:ݢ8̎1.QGyvG|[GX:7h aq 7͹u(jNfsVa 49,ž5][JU c4ɮڭǻٿfAE82Ȝ$rѫucH6DOu%kZ9=z 5"~ Ũ6͎wl Nk<^Ħqh>HilXJsjSfmc(9͟+N`^tٳ5rjG\9B6**7/  OgZ}poC,u!t]}Iy=p<6X2v&9 ;羹|yvұq9SGV@oqLqVnB0q|VD-⨚%8޺9dF56ʭr\Yn#+ٖ,Ź'Yٟr%W'tu lE[Hm&kֱ_ɾWo;0ϏBAE+kY@ ?7Ͳ+SDYng?gIglPĸXՖ޸M?}hк@ M7ݸ)n%ƺn$n=Y)l+ڱ{1:[OĎ&hE|Q,"%F1߹13uX7+?ZhUQbl tP#ZKlnsbZiPVX'Chc{hul?*zqLQ^qBvKPWZۺX r/~lL3Z߷ SS0 n'2ݬf(60:^& U91[(΁VC۷~WqNJQF{8`Wtlь:}5ae'.TewNq9|O':t/jHM.xI^doü)"x-9׋хW)0ëhc\- G-1í#]nqh^%/]> t]&Yyv8â\Ѭ#7j 673K՗,jOi>T,ȝȥrv"庹V.\?'WȵrYR+w$Wr'l-s?s~H5s$!_ s܅sJ"_G%58r=ǹm͕̝iRqdc7Cg|6o0xHnPeJ/}p+ځ?C\DMεl[lg䶣^ڠE?s6%)(U%nFĂX=&*,]r#I,*FF 'CL 1X.6r?Q_[dܨeʠX5Čƹsa m0=\Bţ:Kq'lts?.鸖hUn%ڻ7^$ps&d/Ǽ]M>ԸR޸,ƢI`\a-tXI8˪5c7kXB&*Z{`Mڡ5`t'N݀FRXêՆBM\nq8YJW,ѬՖV.jR:ZPjE(L?TUq ֳ- sZyXDIq z =pav!ge)>[ (>F)~|e+nIM5VPl.S\x7Gg Aׅ R2K)lg ƗRXdn^JV.[9/ /šq96/KGj j1q+Nr#֎?}-~qXJ7x_r@>>bկ{|;nS֋8jCyAt &z(W7ylnג]rX} syyb/e=q7r%zkeW5|FܐM%Ί5Ias'1#d!Wy9|NnEηov z'kVy( r (k^=?{ϜACV97T;~?1~э(Z֗k>Ӽ<қyx+A;iS]^0JqkвU4o|/VxGjҤ C 5hSNX=&w|Dagi/?^ $B\b)eDG13'2=^KTMGYL%1FŽ1Xb([Ģ1h%18>ybƅv1) ֊N.QF븓OdbX-ŒMPsޙF}O[|S G|*$l=艈2~f^8ƶJ՝a?4Dg"D-%w[Np#lEt,P2J{ eÈ3[KZ}lk81X )H^[[ 7c+}7aD]hhZhXap 4_y4~R34j/ņ?t;u3#7Y`WVhy߽3Ɏ97%ܚRAv%9|:H.+Ad}n>WXs B~a&sDDNinLmOW:y;>y͏~h{㷒 rηc5m[!;i_pK7@׾D5Sn|OC󉌱12~vHG{Q6Jab3otx+!1#{JD|~B'Vp2}`)6 wҾ &y!;$ y8z)dk1ځqu)~{dy6d]nho,g*z]~ t q-_Fgh]d+op[E~jemL35}M@TyDy,7㸎/(V 1"zF4kQw+у%gV-O槰zdZk  ~*Ew,RqXbw`+5ZkVB UC,V˴nHPk9:r+BǣDbe0%gZqV%Bi҉83ΰ4wz#֍7 qQ-&-MZlWRغhb3cmt9pFFۉE* D'}%>In! r(x4Exoxo":Y<;0i%,UwS섯"#&tjڂ8uF1C\uzG)ԫĒV ԡlbSP,GhPM:WmZ[@ìvq^ElEJK8J-[*qcpGP_[{ib'…,lC_.oܠ@Wb4_aF7b+trMn{Y<>DLͣ)x?ƜxhmC|oNQ!bV6Dcm#&^s.QxMl::|"㧸h91>11XlR#~'-Z4h<6UUz,W/ s:5xs987^+|Mm+ۍza ;^M7vX%QrܳbTyZ;J_*)ܷepE8Ÿ|QzIb\iu͝ɟ998ɅOD.4'q˚#ȶٞ:-,ZriJ>YXii{1FK[dl) 7d8C86΍ɥrgm]%JH~͹?1dm.4\&ȶzeCVR|#tYk܉|xM^͢Ϳ.`B|(Q׾ Bm2O{X|[N[4_;?- /a\ko6Na\;u~߹!7X6%֌M܍kݺKђIo,%=ZÍ$A?0Kd5s(ZBxt7ںOⱘFahl1:q^ V5*Щ6zUx):C >=`|laǟ-@O違t XW^Gk B omr38 =,%+!ŖW:=_?QmKP[VqvӴMydEufM 0o;4.*]sOǩF[ O휜GK(9U7HDž]^mnSJ)݀X3)+]u~eRrųȥ8Wˑǥqqo3 ?!%9J5wg|-/fEU67KQ&_u%ŨOWw.Y `[_g<,er'rlc~'K[kd#Ț0"Z%AE9"$w=67™?̵9N%?$7Ϯ93wu mɣX@ssjeCwyn<5o|=6zJJ/=m\[ia>O9ILn_rwV/Sy@$ǗqA4EUL&CFMxM1hcqQNbau'Qc QOC̎b?ËoaprAnjyb#~\D9q/x!)/% G!e-ۑW|қ4ޮP M6d Y0b{Q˅9,r+K&!s>Y}6Ad#Hc!mslCͺm%~6\<嵔TF~I7B]vg/ծSٓ~]x0ԟr1?:J'C;O^m|X5M\DMn~^{VMjc^Ņ䟲s>a. F؏\-wȑdm2s䊶˃%!- @N6jIvɃȮٽ%!dly~U.ad<~dֱgs`&Z)dcKm9yW?٫hzwe ax(J5c4~0& nydw'"5\glv= -Է,qU'BL'D︌oG5u5W(),jsT{oZM<c<\w-^)ĊN\K!:$C*e\DtFCloQݺwsb\\bn}~8մQ~-/P^v>Qxh}ttQsb4cThUbn|&Ǐn!*e'MEC";c<9ƚVnG6xmL)]I̋oCw\oq1?FloPB6jP@GXZzBTKeLpWcmĩqQ@ Mwv};R{͎K [j!#,4I>>Jg>u^s'q!i-O100WZl~TRq3: ,^a 9ad,h8Y[R%d9l܊[8,Aن,f4xy2~a}j~xJ0l_$/%(y};A_+ ?/m>_;GB9c^̽^(iҽc_@ MZ } AwnI!pbwKu,Odrw>b^Du>} T{i,;caNs=q}atSl}rW3QJT? U  w"Ί35K5RnmXϛ4 _]ZZH|Wo?FrXZvXi?r6C M⭨{Y^6E'| ,rbMCv7y`'4>"_Ϣi({ǐ6A䪚=GFɦ,cvЍɰLUXy#j[JΕ2c{]|>̿^ܜڛ~^CZE߯h* +.98%ޖ;Z?=͸+((^SPg\=ԺZw<['݋/Uxq>JѐǸdd]dr[re#Ȼ|/>6/>JskS|Z~[}4VKsaW; uʯĿlq=9~e]Qq?a_./&UQt(!j$ øhxhAӨR:u&1(vQnTL"ʣ*&sHqq"QߙL?Ɠnv84b~̌㈦N1QobrE,tG&fD%U1!xwX c1x,=tqK|x>f xs!uiYAlk`2cKg3k XҎţ 7RRezioNUQ3{ʇi;6Hc1ޱ\QrŹ@ec{S\ b/Gٔ^!W΃FV;0Gb)uD6epܛ\+:Ԣؗmr<0mdd܊\:8<\|q ߱~3 m^K eβVo+8hQ-Ze,WO}j톟>;.}.x ^/ܹ盰q$֞|=ǴNA6sWh{xX?DuL4E^De|`*]ǝDm1DxT˨;IQ㕸(1&%QR%ںLc}!ĭ か28&q]BMKL0ǝq'O|mJ</ژ(1q &&cxU`o%;IWςXj>qW^l 8qpGIsK6_r 7\u)*|Vvlz^Z4Do) *AZ翪 -|¬Ah;KE3VGeowVme|1GwCndM!fYrd;pch6dPoarAf6F^yId!W8w 7޻Gdo'Xo%tXg][V?>uiwN?_vRU5Sv;~ԻDzNTFf}VG൩E> \txH:q'a}<}Dy|x# D1VxĂӈcB\JtJΦ]x*ƭuw(Tq;QLuIlt #c|Q1"ڻx6!QSLuK")Sc|L$JMk Lu¢^}5XwXuj`Q?6onGL'iqrXb mh#rYfa%j0_g󑜌\n<rgIy JSUհբ:dք _odvi]9Tܦ}a>We+L¨5qSwP6岊ⷶ y:ٜ?.&9k.4\74 07).nګqԎY4%Ns;kOޤEs^ٔq\Ez5A Z%h77RA6cjo9;!gA_zNe {~nqxJhF|-;wS,H|O1n[)Gt܄{bR1c4qF&sGL ^/ȧ7OّGϼ,9a~_'7ԙE6Ag'QK[ *,YGcKBɊ+)9pQ/^-4R&*Il~9|/7)z.78bOi|vSOkںirpqqq>o*Cۛ/ 7yWbS ֱ5ڷEuK%ټ" ?)Yrgɟ({jv TkPmhwF}J%V|f`ٛJCEY3Ln;:wj^{CꗩыLu?=rja%-A-~܀udWE?dU8)վoxg>C[Am{+v9dofi 47So"Ӹhc1+6hc$ccҿN"ŃcA#/b"?Ə1OC6z?&[zm60X3H|@qM$DŽ 8*%ZId\Oq&aTL-z3bWuoq3ecc6׉̃h`9XXU1n;5PqI\Ca/K6NN*}m$p6(o>QFg9vmg qFF3 Q-4m\CӏM7Sw)_o8'lܚm|['ְQR}+Y`-3_&RQپ~fVdJ:_Z-cϮz/x`si>n:^r4~xи}V~(lrmu@ѿ3◄mz]W:l^<7ztγqXƚ}}<]5 }`c?7o?873>˩{gP4'$F}DGϣ$1Cw*zQl!To[ڶ#qsy*g*CP,FbPDaQɞ%QqKiSqj͕KynMiܹKY4nUu/>U)nu#cQ#BA tcrF?{Q7asnt4%6FpkN)>J6!O;/Ķqk|⤸ Z~E|-ɇ{'čͷ.GNg_; *g̸x%oVxhsm8-VjShn-㥸".'GoJ~hௌCm<@t)q+!G<ﮘD.$ގ7^(ĽD{:(8x22=01nxnyUz(s1x4&ؐh[7J:O317q0ts )&cJL&6CcjKtbXxFc1*nMFx:Fĵ.7 ?6Nxp-3+]wNYR|æ1ԮXXey+Éǡq86Wj,'CFXf6AS\g*muڠ+.w0Qx3F k|WM^SY׎͓QWXf%JUEs@яǰ$h񈮖FcK~#RUN>ROi43(zn'^[|StơM*'B,5&ǶĘk?1X1hvQDsSG|n%1I<ZtcJ|c'5mbr܌Cq1x,Tx;cbX11M087΍1ۡ(c?=qQ6&j4vc2['ٌx< x01ą9K$F_c [F}߳%|wgR_6W&)v)LV+\:V v5LX=VGq‚F (OZ *)'.jU+E(=vi8_[T2R7%]ߧ{6|?:Zoiw{xAq-J6}aL?s{ck? ./t 4_0vB|iO}8QkOҥVՒG |jR70ٻ]l' r^^OM.!wAyyA%r+drpIs/ "kBu{ɟr~In[{ rrn9/\GHrSh46wLFy($Z9fܝ,n~Gvϟs(͹xr>Jvߍ$:|;Ke닫%([ݫ }?Z{(=>`_S*=7~ K,+'j3;Kٹ_IǶd gqLޛₜp~nۓ]]B9r 3SBpvbW؊ϓgHHf̜9{]F3 h2o (BwQ8/{q߅1Yf(lg'Yw>' 1mI{aO;D ^ OdZ,Q#l( ; 6:|JZ:FEe  䣱lR /#2yƑ8dL e$s,oe @>`$nA2Hɼ \c ʄq4NO)2Q@C\a A@fH C3pOJ4"!HKs©\%tVHsN(MZ2cd ?AQ)ݨ}bkt3퀣ݍ]!eԡX#7 4jlO3P 8?p ((sL^!hMn3ۖw}qи $+X~O#G_" ]ó LzǺ\@T/wqGg31dBI 4ֱAim 5A[G{c{@Kh}^z&G;4h0 &}M6@֠i:ZY;Ok$=A+jF&Q@[͢-AS2c@=@ :ZZ dmD4h4[8&qP3i1:4RS4[%h#K[eEUmz[oisЂZɠkuf=4;zch>}}lΆxBGS d.Wڡ}s-G@6e'@B*$@hG@P,2GleNk.~ FfomO70I`5 xw[޷ACo9޻KxY~4X,[ J_gN1Ƥuh?5?^չD.eA&;=huzHz V =$uHNHw+ZOz祳> et>N P D! hH1,fN$P@?IY \U4* D(i$뺗RT2l.4RMzU6@$sI=H3jL%Fopz6N#O]Lf`[6ЗZZHZHZk}*Ys~(]RXVߛc^1pQМ-oi '~ DpQ_a'?["c;Dq 85)@V`/h11ЌZK@Pb4i3 yhb ͞#Y8ia8巇%lClۀB2[vcj@TSt0C<23!wl%H˃~@zq %iiX2|Nۤ3:[ 4־+G#?~rZY=?Ӵ'.^,Yw?*GIy%qpG(@jM9H鮝x+0]-=u)PB3 8#@? ld3@JY.dP`}$U Igs2J_&xm7Ixjߒ~ yh@>:Vh#R.N=OHS Yh- $y8Nn֢]{ ݴ@XF6cfi)Q?5@iJ5f55)= L#I?IDAT<tfZ@ n *EH4H4p*X]㥑iUs[BYbKHy 8?SSݲ@&!kW /'b'@DMmznOЏ "7M0|c\G*3< ޽b$TxZ]óL{Ee+}ej"p8@{h4@'0`ˈ].7%*҅ t). Q@Jt%}|tKɨnr`4@|,tbPj\8pQ HMY&Rd//0E.]($_tMTJ6YBz~G34Lρ @N@)IiyhpC')$@u.]A%H  zMj @3@hBAߦ4m aqxHSm?K=|Cd;/ut9‡B2k-F2bXCˏf=x`G@>@YL+Uu~-,}W6RRz],2 VzIWbT}MQ(3D8ܴ"y8XRN;8',$IO$HЧFm`ni *: P}W ;Jը x @0ۨpu+@Mi_c:P d DSPp  ;5YDV' eFs޼$dS}oe{dT ʼ n|)w8ͯ=vcY)СcA\ ]¶eJ'JV+=!Vl2SdA?-o[̑cyE!.ig%ú#Lk~w)p 2>tඌхcwrhXWq[j!3Q; Tն MT7.": 2 ȣ;1 +=tI|  QKVS#V'<$Jwu< \vRGzqәT&pM{HPDKI'-$jk}`?;1tfM_2QZYC? Ir^J3- CU 쀿6!@mjKksДZ"D)@.I[`= (I$P|Gjy8xe2^@#|@jc[@<`3ϛ#2 Pj_.! ⹒on|EH*|oɍZ2Q6N IOŒF B&n\ZŽ7Bs>e˴c%Q/ HO vgSr87ݠs`ڊ@)C)l>wdG~݀((CGz%ȠSMD@qJST'H.X! 5ѩ@2Mv)-RZ@Fm]d@FY^)L] Y;H(m7PYt>8IH1-lҶdIgOk& TN 0 ПJ?RhI/yAz^PO4%HwAgdcPNb1x 8x*up_cnV4|.ЛMۍ|++e@K 1{:` 2D-uV W?؟N_Y.5#tE 2+8P}Gs%Zҍpn!p :f@[h}mԣ$(H\c*:וP@i D'|[:W*p"  PFA:Q_]&ݨԉL@dAC/dB3G}(C;* dTWB]gݷO@M*$t4<P$@7@=B9>MY@!@_Ĩ6A>֍R'IYji^נ:`Ȭ9xʃ؀WP`rc@aPt0G֓ӷA2׈rB%wj ۱RB (cd@s54`}qeI!|f{ͿgYJ6o{կ\b:ߎ wV@n^8{RHom)u_Y`<^bÏ`{09 9́S̠JF4 y,AӖQ@7!Ӂ p2G68&h #6rH }@P@ G|K2YADy%'ܣ@B)ԕQΒQ8Ƌ@;"pY s@>4$ c9(GLbH ΑH]v 6oH= | ̗Z4 'qydΑ4q(Y32xY\X;+$~']((%G@Z,ߘ$W=ޣ^=0;u=w6ĉ\4n>ZCovhW.<])I?:}2ĭ]x?.1vѪJ/j釀f.yTT枎E MK*J_t2R]x[' AW}]&jk ܥ2҃jcmBiæ̀RT 2Hagu a@"Y hr|CK4c@ mTnq(ʛ}$]+H%U*Vk=KuCd!'V!Bu yq l>pM *v?DN9Nńa*7w}p;k<|)ǐi(ܝv2emQf0'1Z=Ql#?CF͞Q޾;Fڊ/~֡~2a7<#Ǿ\vܸM3sj~9u ƟB[;s*};~jzq+@ՉpK`dzDЫ\$ϡ_#ћ&|WSf Jc@m`o5)apݹ(3%ǿ}Ӏ/:ߟiȣH,d{]ctzl9c>iK'#\pJk+`GF@}2IK+<@dx[6_]9X @VnoX?l yHJ a,mٰy6oh  ґ@Zi A*Nf\2ju*y Dhp,}JnX  0҅f@V8 )@QD$*Kw`?tHNkNN iܗI>ZJ=@ "0pH&0 Ih5t"זD5YECy_(mU\3q=*ePQFU?1@irQHjn;bhL7M}=_8AV'1Ȉl=+<>91X=Ȓ%ׯ2`v1K8j;ᇛ\^af!OJP>h,r<. b> l A.>fJihKw*̀K}L.>@jW i$SO~`ڴ3P'(Rct2KE-` &M:2 )4fU@ , t>#=`&N3];CS_'vB@=7RK ;2 ;`9ԡ"5|H9ВRkk/s_(>%x8k_G^Lg>oÅFn/FyLtߎ|/ЭWw_X /.9L_p2nm_g(wl~_R+WZB xH;'P~xu0pRcH]o6n<@6k4u Қv¤:8Cu4\s /rza(U:mޣG Vu3CAx, EAMmyU^pݤIq o }jL\g'Ja'; %3hxf_׼wHW{ ! uS3ߟt?|evΛ.5un$IÜ`CM+3F Ԗt@ݼWQ W4KHz,m'Q 'p"nGMŮgwͤ6|LZvѻ?=mPgt"bkЄp$=^EsrU'Cp&R7g dN*A!r@!*})U^.:!R^@AS@sjS P8IyM5'%s`1j7Ba*L+зt"p6<{ҙ€҅p<9$ȓф@o''5@IPY($HC^22S,W9g9e6}@Nb) GAQݼ !& Ib|[28f. ʎ !'RpXGh3la|ofO'd 4nKGw<5rPUPϬ 1c}uЧ$U< yly:84YRxI_|+:l:g uba, K xW7A[9H RҘ43& +:ѕ6 ZJ[`n )KK`:9h 2 $ Y(6@;`-6i 2̤=0Oc5DKEi^4d^*@ d55]CFC% Q% 8 5߃p }a+v8b^3q3 xD߉R9=`'sK|@@q2b !}i+Ӂ'>v{3̿s:Ӓ JgDAVI_Z/,yKjPhRiZ<<-~h.= r4Ku h L"mAJtt9 igtg6ȋ7Sf, ߄2  2'Ai$K< iGC","lPkpet_Ԟ>T GKaׯ~r[.ѷXi(8O!ǘ}MiNs (J9=<+OJ'{D: p`G&BAUN] J!4 !Nn ~'m+mFm;&xƝSp TIz"Sd dq$yTҐ2rOWn`ƖgvN-g4ZY\;l3u+ZGGP;+! /80g\s ?@aLiPf'AЃ@=OaeTֲ:&+PQhFc8iX#@ l2lW<2 0dIl) o2M^ yG( Cq׽k!s(jB>o{0(7Wv!dזzM.A < MVOQd1E+pꂄXz%AZ0@/'@7ON) 26&hƐdL/ d_gA,<|]m( )TzH 0^/`1zDPWoNiͿHoW;MadVX.x苃?/2ݥ;d$N)׀>9؁ *pd}ӌ:Y'ʫ@% |4N$ĭX q BK*9ÝvlŋZDezP&ݽ-²[ndV1"l॓@'+XR[L|>YVnohj/:JR pPFX;M5Oriҕ "= yAV]UeOxz !7b~A3 'DC 76O0˴@`0 RdL~/fc";@ZOH ;hxfm6hj2GfV5.klnX t )h,a/ɠ}u$) VFcMl&5 >-$e|/33^}]Hp{…ݠ.`?S,F0ςOM͞,E "u;ek[EwCvgզ_u]V3+&28* M3GЈzuVAHF |0Hrd񸲠+4|z!d#@Vlh(A撑s69t xe, Q)2AN`yR/\$,,0OFÓdra<~ЌJY"ӈ(Ksh )ns3 !\Pa:nv|?w`=? =9;w5:1^>ZE\K Ez^ x /A0K]:H&PPɳ!%k'Aҫ[O/cB@y P?q{,_;^ `<6b ;2mpuRhqX;su}V7|ӴˮLeo]R>[~+sS𝯤uV`#_e 'U/KcRg:!+dh7$:P꾥6^^s,]ۓ^I4Ȼ 0<"@&%y_(\2!j1RHa*^18poR,4'Iv.x (5ͮa ÷?D*+?[~/'c?\l6oҋ$<.6;=ʣXL@m79 Axt^ UNez$^Bx[}Cg;0~B'eAwK ۿE;92`#@DRfc+Ps2Hy2RygC՗ Sn;y[Lܾ$5r+>M3OШg7-Z.,U ;^R$;_:߹p 渀ER(I2 0>@.B,ba2%9 (((xifV+K&Ӂ@ 209i 8Be oL2A8d^Vph=Ml@GcInq6̂B#] Ils_qkqЖI!@(I7x:XvrH.-OMygKy  _-K8tW0 'N@I& p8hmFxxO3C-Ņ ើg;pGlްQ?=5R@Ji[\)nV<;b}IN"!ŷo5dGPn("M<w'P*l|82H.cP흃>7?X~᯾|eeH|#8 q@>R)kxAmg %a }F8qK''~mw!`#d8ABP$)ye>(Hޣ.`H`5}]̠?oM_n͔n>؎t+9 s sfC< m@A fp3#"a@~#3ySJגڂ;~+OO 7$}V>~a ,)pC侽ZĘQ{2Klf;,#mWY@SN]`M;6(,X~S#n,,HUVvZx[hJHd,<["dV2~}k%+j^!=օn=V٬ ـפ)mޅ_s0S&!,(k޷"ImIj׿?b.F@6W`Uc=+{_Ùףg&{5 ^ bԒ^IwdGz{_Q i; 9b5?&@!$D29]H-trl~3W3쎼rWY5\&;+F^u 0i,YF6nk:Y6WО%<&jqx _ IpƯZ{ӥ6<Ǐѵ{9z 04y̅?~.xC spٌXϿ;gj bjOyJ~FIUM8Բ .5Nܣ?j6ԢWٷ=~QZJGN1S^`*Hg "Rό\(5юn(Jwx[ lM3A 9f'`oeDVu٦={{!c&Nm٘5)?hQS=ΓIFp]w۟H'ojTWt(0^JW_k8ZM/ԭZ"U+ \=sv|r̯h3,xnWػ%e\3 Þ|0λK= LNu 1H e2/i=N۾l萗<0}`ߔ.>>.a?"P@yIq_R #lpVH赡dRuH-li|diVlL%˟B'TVo[!qo# -X:7Bl~M NT^Y.ő͟`eњoM$hL!+!ܻd P+ rNHFq\7꥟5'Ǧ#;#]0>5-(kddcڤx3ncIKo1 ;CM^a" E"xH!}js,H5WV(nRUC+\šu1BtbI`z.E/ ^s7Â/e]2L74p<{Y lDu{w{gcg`6rNKj gdԴȲKf/"hm?3fUuP90R4)"'T'23G/,~y< O_GF"Ko;x;ؿ,B_tkt-Y2FDQ]hxf4ٰɌ8sꪉ#Mmn_h$K[ГLT06'񯂷Ik1x&aCx%)KrӼ{jRNwNz 8mij\_ZIM\@T cd5PHJ- rR*!Łl{ lZے@Ua @ :N X ~.v'>nZGtMI%iiG. =~ӣw>-dgK33l`l{W5t )q]>|VS:zPe-le(D[#đbrr|'vBZ1o:T|'$>B&o<0.!p#!rwcYϮ'yPI|F2QGL{]w}\R\;uϟI1#\myd?V#;Y˾JGIG_iVm>qL#7'Z[kml-));ATNXˀvI7g.h fcw=$@g1{!us\{ߵ0[t d_5`oHnss@#i{Wl=@^bx6 dYfn"٪oOH]^r/q ~.kxP6;hܞ6 dk< {v f"S2i\dbXn=Ǫ=Liΐ?rp30Ip%GGF&ԗ$x.BL Rd+IhR40J튵h/.$I9w(bpd!dv?7NcK_!rq$!gk )g:G{Mq? /<*1d)3ӏ,٣pZa_L_9ۥ-0^BG ,8m8HO܆ _HGdγH*ǀdpۀXnJ{7p D~"H*A<HJ5rC,&u1yI;[rZK.F7>0mшDzOV`cr8vr=s6=‡>b /It$rR^4O!y\:! !Y6\`ܣ6d;gqگcF…m0jV;j):P5b7%) AosSaJC`oØw̳ނ8r ޕsMa3f&ux}DrTcaJGxIi:̴WcEtw硬u4( E& z7QS,T;{C&Aw =[yEƹS˺$S~#m{bHwm:G3KJiw6_ݴ7 qҌ|)ߓUM?K~r(^nk{ˆ! (.UZCq}2]_9^;yqh.:+X,3lٌw-my>o4Nvp(^҅ufU|I]b>ɇfp9KzCǧh sbw+!}X b#/쿼$IM~# 6|0$Q3:Zo8sIٖ],ArA6md;ҭگ| q&8bR>a&/wd=gaܱwK`:bX~%Yqڞ}ÝU+knEbv)bkDIhOrg..8FŨw[ [ ),ON~}AY bJ֍;# rVmSa0J w,*l%RK_;AÕ-^aV:!H|xG! %; às-ڙAyYcS+vnQ*%kLБd&Q/a#Nl$gIz.W4tHc4Lι(ļбX,ƣ"VMjdIr#ɢ>Ɩ7 {ҷXa 7^|%kKr@wK]A”[AVX,橬D1hI Oenb%q-Tꑕlu'Ks(&$AsmƄeu!gX,7.UF~D13uEiX,%M;|JtGyvl6E>Epq+5Q#IdAuNiBTPrGz-m[̬qT{~+]AbX# Gu Qc4vc:;~SaǃVlج߄dȵ`^ i#{xެm uPX,?A\[]loMf[RCNIz5G7ڻAj 3Wm;D`\酆.X='3s9~ Rܮqz'ވ M{\',W_'% *_VynbD<{6zDW߰'X,?Up'ӕOhl7ͱ:&ʨI巵4xaYIF>7cb:_ϞX,su۪A{N-C Oy7yPnIm* *N ZpfL.;VDcn]'髣=rbEL ;IЖt֎c ?NՎq xQv)^ۧޜu|}M02̚{uX,_nf Qy>S4m%8)H$~o*> #r!:ϖdٞ)gL9+X,# Gk[/<텠~e&gwƋ7UeQQaOZKY bEM:XnRVNxy3{iNXGzkMQl7!LB%:55֣7˦up?uHX,_TFyb)._'=7=Ϲ Lc(5ݟZ #_rET H/bX&|, ES l%Kj+/s.gZG$f=R2d #Rx!:;bXfe>7z5oiu9>z#V6pt?#xwZG[[atc<~1@UB/bXU>*uk^xlq1nn.DxH82RPyC `CxX`dnWXQ#bXyGv^G;/skO:鼾PpZZi1 !Ƒ[ ܃ u?X,?EKEϻ]..7ǮpqܤA9Bn`4`B;BB<b:򣅹]/`ϩUCȋܘ%v2(n:?!ܡb.l(Ɣ9G,:byĨ]o8CQ^(H]m!q|#p;!3p*(G?>gzWzjŏ#f=9:X,˿vbT,{DyCKԦ(FNwq) (Ov?} uqTV`X,Ӵ+vXC烩|)Gs*~Xؽ6 vtg$ͧ\9r1˲ bX,Uʣhk(r.I.oalA W]zJO e͒Kx'+[1:WK`9k^bX'7]]{׬awH\t C"^$7(lQ ?P ̖1Re֋]l n>rK:iZ,3L2\ܙ1|Z#oߦ8_SlZ?@&)$#Qva?|qsD$szv!b,cRsY!k9cϻUIbX,_iTGu)!4w-yq=& 8W K]+m|swg :Ou[ruF|)_ߟ`X,LJ^HyVʺ iOB4s}$~ FEp9KzRl?k &jLHij&6׈5^s@'p6SS/bX,+ge>Op4bg&I쒁|hz\FLQ8vfC5@ {3̩!lz?q2mnVM;9B7JWg%b|i1WdJY@[UmHTźEQ>WF, 9%k>L59NլL!vY~)Yew;z- |<65baf/)z@m}r|ca$. |_Н0ueMeblc)mJ;643i,bX,bX,bX,bX`c)ȼIENDB`pineapple-pictures-0.5.3/assets/icons/app-icon.svg000066400000000000000000001307471415740431200222110ustar00rootroot00000000000000 image/svg+xml pineapple-pictures-0.5.3/assets/icons/go-next.svg000066400000000000000000000020541415740431200220510ustar00rootroot00000000000000 pineapple-pictures-0.5.3/assets/icons/go-previous.svg000066400000000000000000000020561415740431200227510ustar00rootroot00000000000000 pineapple-pictures-0.5.3/assets/icons/object-rotate-right.svg000066400000000000000000000023311415740431200243430ustar00rootroot00000000000000 pineapple-pictures-0.5.3/assets/icons/view-background-checkerboard.svg000066400000000000000000000033551415740431200261760ustar00rootroot00000000000000 pineapple-pictures-0.5.3/assets/icons/view-fullscreen.svg000066400000000000000000000031051415740431200236000ustar00rootroot00000000000000 pineapple-pictures-0.5.3/assets/icons/window-close.svg000066400000000000000000000044411415740431200231040ustar00rootroot00000000000000 image/svg+xml pineapple-pictures-0.5.3/assets/icons/zoom-in.svg000066400000000000000000000021601415740431200220560ustar00rootroot00000000000000 pineapple-pictures-0.5.3/assets/icons/zoom-original.svg000066400000000000000000000027531415740431200232640ustar00rootroot00000000000000 pineapple-pictures-0.5.3/assets/icons/zoom-out.svg000066400000000000000000000022021415740431200222540ustar00rootroot00000000000000 pineapple-pictures-0.5.3/assets/pineapple-pictures.rc000066400000000000000000000010031415740431200227640ustar00rootroot00000000000000IDI_ICON1 ICON DISCARDABLE "icons/app-icon.ico" 1 VERSIONINFO BEGIN BLOCK "StringFileInfo" BEGIN BLOCK "040904E4" BEGIN VALUE "FileDescription", "Pineapple Pictures - Image Viewer" VALUE "LegalCopyright", "MIT/Expat License - Copyright (C) 2020 Gary Wang" VALUE "ProductName", "Pineapple Pictures" END END BLOCK "VarFileInfo" BEGIN VALUE "Translation", 0x409, 1200 END ENDpineapple-pictures-0.5.3/assets/plain/000077500000000000000000000000001415740431200177365ustar00rootroot00000000000000pineapple-pictures-0.5.3/assets/plain/translators.html000066400000000000000000000006451415740431200232050ustar00rootroot00000000000000
  • Chinese (Simplified): Percy Hong
  • Dutch: Heimen Stoffels
  • French: J. Lavoie, K. Herbert, Maxime Leroy
  • German: K. Herbert, J. Lavoie
  • Indonesian: liimee
  • Norwegian Bokmål: Allan Nordhøy
  • Russian: Sergey Shornikov, Artem, Andrey
  • Sinhala: HelaBasa
  • Spanish: William(ѕ)ⁿ
pineapple-pictures-0.5.3/assets/resources.qrc000066400000000000000000000010641415740431200213550ustar00rootroot00000000000000 icons/zoom-in.svg icons/zoom-out.svg icons/view-fullscreen.svg icons/zoom-original.svg icons/object-rotate-right.svg icons/view-background-checkerboard.svg icons/app-icon.svg icons/window-close.svg icons/go-next.svg icons/go-previous.svg plain/translators.html pineapple-pictures-0.5.3/cmake/000077500000000000000000000000001415740431200164115ustar00rootroot00000000000000pineapple-pictures-0.5.3/cmake/FindLibExiv2.cmake000066400000000000000000000072101415740431200216400ustar00rootroot00000000000000#.rst: # FindLibExiv2 # ------------ # # Try to find the Exiv2 library. # # This will define the following variables: # # ``LibExiv2_FOUND`` # True if (the requested version of) Exiv2 is available # # ``LibExiv2_VERSION`` # The version of Exiv2 # # ``LibExiv2_INCLUDE_DIRS`` # The include dirs of Exiv2 for use with target_include_directories() # # ``LibExiv2_LIBRARIES`` # The Exiv2 library for use with target_link_libraries(). # This can be passed to target_link_libraries() instead of # the ``LibExiv2::LibExiv2`` target # # If ``LibExiv2_FOUND`` is TRUE, it will also define the following imported # target: # # ``LibExiv2::LibExiv2`` # The Exiv2 library # # In general we recommend using the imported target, as it is easier to use. # Bear in mind, however, that if the target is in the link interface of an # exported library, it must be made available by the package config file. # # Since 5.53.0. # #============================================================================= # SPDX-FileCopyrightText: 2018 Christophe Giboudeaux # SPDX-FileCopyrightText: 2010 Alexander Neundorf # SPDX-FileCopyrightText: 2008 Gilles Caulier # # SPDX-License-Identifier: BSD-3-Clause #============================================================================= find_package(PkgConfig QUIET) pkg_check_modules(PC_EXIV2 QUIET exiv2) find_path(LibExiv2_INCLUDE_DIRS NAMES exiv2/exif.hpp HINTS ${PC_EXIV2_INCLUDEDIR} ) find_library(LibExiv2_LIBRARIES NAMES exiv2 libexiv2 HINTS ${PC_EXIV2_LIBRARY_DIRS} ) set(LibExiv2_VERSION ${PC_EXIV2_VERSION}) if(NOT LibExiv2_VERSION AND DEFINED LibExiv2_INCLUDE_DIRS) # With exiv >= 0.27, the version #defines are in exv_conf.h instead of version.hpp foreach(_exiv2_version_file "version.hpp" "exv_conf.h") if(EXISTS "${LibExiv2_INCLUDE_DIRS}/exiv2/${_exiv2_version_file}") file(READ "${LibExiv2_INCLUDE_DIRS}/exiv2/${_exiv2_version_file}" _exiv_version_file_content) string(REGEX MATCH "#define EXIV2_MAJOR_VERSION[ ]+\\([0-9]+\\)" EXIV2_MAJOR_VERSION_MATCH ${_exiv_version_file_content}) string(REGEX MATCH "#define EXIV2_MINOR_VERSION[ ]+\\([0-9]+\\)" EXIV2_MINOR_VERSION_MATCH ${_exiv_version_file_content}) string(REGEX MATCH "#define EXIV2_PATCH_VERSION[ ]+\\([0-9]+\\)" EXIV2_PATCH_VERSION_MATCH ${_exiv_version_file_content}) if(EXIV2_MAJOR_VERSION_MATCH) string(REGEX REPLACE ".*_MAJOR_VERSION[ ]+\\((.*)\\)" "\\1" EXIV2_MAJOR_VERSION ${EXIV2_MAJOR_VERSION_MATCH}) string(REGEX REPLACE ".*_MINOR_VERSION[ ]+\\((.*)\\)" "\\1" EXIV2_MINOR_VERSION ${EXIV2_MINOR_VERSION_MATCH}) string(REGEX REPLACE ".*_PATCH_VERSION[ ]+\\((.*)\\)" "\\1" EXIV2_PATCH_VERSION ${EXIV2_PATCH_VERSION_MATCH}) endif() endif() endforeach() set(LibExiv2_VERSION "${EXIV2_MAJOR_VERSION}.${EXIV2_MINOR_VERSION}.${EXIV2_PATCH_VERSION}") endif() include(FindPackageHandleStandardArgs) find_package_handle_standard_args(LibExiv2 FOUND_VAR LibExiv2_FOUND REQUIRED_VARS LibExiv2_LIBRARIES LibExiv2_INCLUDE_DIRS VERSION_VAR LibExiv2_VERSION ) mark_as_advanced(LibExiv2_INCLUDE_DIRS LibExiv2_LIBRARIES) if(LibExiv2_FOUND AND NOT TARGET LibExiv2::LibExiv2) add_library(LibExiv2::LibExiv2 UNKNOWN IMPORTED) set_target_properties(LibExiv2::LibExiv2 PROPERTIES IMPORTED_LOCATION "${LibExiv2_LIBRARIES}" INTERFACE_INCLUDE_DIRECTORIES "${LibExiv2_INCLUDE_DIRS}" ) endif() include(FeatureSummary) set_package_properties(LibExiv2 PROPERTIES URL "https://www.exiv2.org" DESCRIPTION "Image metadata support" ) pineapple-pictures-0.5.3/dist/000077500000000000000000000000001415740431200162745ustar00rootroot00000000000000pineapple-pictures-0.5.3/dist/appstream/000077500000000000000000000000001415740431200202705ustar00rootroot00000000000000pineapple-pictures-0.5.3/dist/appstream/net.blumia.pineapple-pictures.metainfo.xml000066400000000000000000000017451415740431200304700ustar00rootroot00000000000000 net.blumia.pineapple-pictures CC0-1.0 MIT Pineapple Pictures 菠萝看图 Image Viewer 图像查看器

Pineapple Pictures is a lightweight and easy-to-use image viewer that comes with a handy navigation thumbnail when zoom-in, and doesn't contain any image management support.

菠萝看图是一个轻量级易用的图像查看器,在图片放大时提供了方便的鸟瞰导航功能,且不包含任何图片管理功能。

https://github.com/BLumia/pineapple-pictures https://github.com/BLumia/pineapple-pictures/issues ppic
pineapple-pictures-0.5.3/dist/net.blumia.pineapple-pictures.desktop000066400000000000000000000013251415740431200255360ustar00rootroot00000000000000[Desktop Entry] Categories=Graphics; Comment=Pineapple Pictures is a lightweight image viewer Comment[zh_CN]=菠萝看图是一个轻量级的图像查看器 Exec=ppic %F GenericName=Image Viewer GenericName[zh_CN]=图像查看器 Icon=pineapple-pictures Keywords=Picture;Image;Viewer;Jpg;Jpeg;Png; MimeType=image/bmp;image/bmp24;image/jpg;image/jpe;image/jpeg;image/jpeg24;image/jng;image/pcd;image/pcx;image/png;image/tif;image/tiff;image/tiff24;image/dds;image/gif;image/sgi;image/j2k;image/jp2;image/pct;image/wdp;image/arw;image/icb;image/dng;image/vda;image/vst;image/svg;image/ptif;image/mef;image/xbm;image/svg+xml; Name=Pineapple Pictures Name[zh_CN]=菠萝看图 StartupNotify=false Type=Application Terminal=false pineapple-pictures-0.5.3/pineapple-pictures.pro000066400000000000000000000055111415740431200216660ustar00rootroot00000000000000QT += core widgets gui svg greaterThan(QT_MAJOR_VERSION, 5): QT += svgwidgets TARGET = ppic TEMPLATE = app # The following define makes your compiler emit warnings if you use # any feature of Qt which has been marked as deprecated (the exact warnings # depend on your compiler). Please consult the documentation of the # deprecated API in order to know how to port your code away from it. DEFINES += QT_DEPRECATED_WARNINGS # You can also make your code fail to compile if you use deprecated APIs. # In order to do so, uncomment the following line. # You can also select to disable deprecated APIs only up to a certain version of Qt. #DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000 # disables all the APIs deprecated before Qt 6.0.0 CONFIG += c++11 lrelease embed_translations SOURCES += \ app/aboutdialog.cpp \ app/main.cpp \ app/framelesswindow.cpp \ app/mainwindow.cpp \ app/graphicsview.cpp \ app/bottombuttongroup.cpp \ app/graphicsscene.cpp \ app/navigatorview.cpp \ app/opacityhelper.cpp \ app/toolbutton.cpp \ app/settings.cpp \ app/settingsdialog.cpp \ app/metadatamodel.cpp \ app/metadatadialog.cpp \ app/exiv2wrapper.cpp \ app/actionmanager.cpp \ app/playlistmanager.cpp HEADERS += \ app/aboutdialog.h \ app/framelesswindow.h \ app/mainwindow.h \ app/graphicsview.h \ app/bottombuttongroup.h \ app/graphicsscene.h \ app/navigatorview.h \ app/opacityhelper.h \ app/toolbutton.h \ app/settings.h \ app/settingsdialog.h \ app/metadatamodel.h \ app/metadatadialog.h \ app/exiv2wrapper.h \ app/actionmanager.h \ app/playlistmanager.h TRANSLATIONS = \ app/translations/PineapplePictures.ts \ app/translations/PineapplePictures_zh_CN.ts \ app/translations/PineapplePictures_de.ts \ app/translations/PineapplePictures_es.ts \ app/translations/PineapplePictures_fr.ts \ app/translations/PineapplePictures_nb_NO.ts \ app/translations/PineapplePictures_nl.ts \ app/translations/PineapplePictures_ru.ts \ app/translations/PineapplePictures_si.ts \ app/translations/PineapplePictures_id.ts # Default rules for deployment. qnx: target.path = /tmp/$${TARGET}/bin else: unix:!android: target.path = /opt/$${TARGET}/bin !isEmpty(target.path): INSTALLS += target RESOURCES += \ assets/resources.qrc # Generate from svg: # magick convert -density 512x512 -background none app-icon.svg -define icon:auto-resize app-icon.ico RC_ICONS = assets/icons/app-icon.ico # Windows only, for rc file (we're not going to use the .rc file in this repo) QMAKE_TARGET_PRODUCT = Pineapple Pictures QMAKE_TARGET_DESCRIPTION = Pineapple Pictures - Image Viewer QMAKE_TARGET_COPYRIGHT = MIT/Expat License - Copyright (C) 2020 Gary Wang