pax_global_header00006660000000000000000000000064145157232310014515gustar00rootroot0000000000000052 comment=fd5929bc47f0e88fac930c881141d22fabe1e81f pineapple-pictures-0.7.3/000077500000000000000000000000001451572323100153355ustar00rootroot00000000000000pineapple-pictures-0.7.3/.github/000077500000000000000000000000001451572323100166755ustar00rootroot00000000000000pineapple-pictures-0.7.3/.github/workflows/000077500000000000000000000000001451572323100207325ustar00rootroot00000000000000pineapple-pictures-0.7.3/.github/workflows/macos.yml000066400000000000000000000004561451572323100225640ustar00rootroot00000000000000name: macOS CI on: [push, pull_request] jobs: build: runs-on: macos-latest steps: - uses: actions/checkout@v3 - name: Install Qt uses: jurplel/install-qt-action@v3 with: version: '5.15.2' - name: Run a qt project run: | cmake ./ make pineapple-pictures-0.7.3/.github/workflows/reuse-check.yml000066400000000000000000000005151451572323100236540ustar00rootroot00000000000000# SPDX-FileCopyrightText: 2022 Free Software Foundation Europe e.V. # # SPDX-License-Identifier: CC0-1.0 name: REUSE Compliance Check on: [push, pull_request] jobs: test: runs-on: ubuntu-latest steps: - uses: actions/checkout@v2 - name: REUSE Compliance Check uses: fsfe/reuse-action@v1pineapple-pictures-0.7.3/.github/workflows/ubuntu.yml000066400000000000000000000012241451572323100227760ustar00rootroot00000000000000name: Ubuntu CI on: [push, pull_request] jobs: ubuntu-22-04-build: runs-on: ubuntu-22.04 steps: - uses: actions/checkout@v3 - 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@v3 with: name: ubuntu-22.04-deb-package path: build/*.deb pineapple-pictures-0.7.3/.github/workflows/windows.yml000066400000000000000000000014111451572323100231440ustar00rootroot00000000000000name: Windows CI on: [push, pull_request] jobs: msvc-build: strategy: matrix: vs: ['2019'] msvc_arch: ['x64'] runs-on: windows-2019 steps: - uses: actions/checkout@v3 - name: Install Qt uses: jurplel/install-qt-action@v3 with: arch: 'win64_msvc2019_64' version: '5.15.2' - name: Build shell: cmd run: | set VS=${{ matrix.vs }} set VCVARS="C:\Program Files (x86)\Microsoft Visual Studio\%VS%\Enterprise\VC\Auxiliary\Build\vcvarsall.bat" if not exist %VCVARS% set VCVARS="C:\Program Files\Microsoft Visual Studio\%VS%\Enterprise\VC\Auxiliary\Build\vcvarsall.bat" call %VCVARS% ${{ matrix.msvc_arch }} qmake pineapple-pictures.pro nmake pineapple-pictures-0.7.3/.gitignore000066400000000000000000000002011451572323100173160ustar00rootroot00000000000000# User files *.user *.user.* # Translation files *.qm *.mo # Generic Build Dir [Bb]uild/ # IDE/Editor config folders .vscode/ pineapple-pictures-0.7.3/.reuse/000077500000000000000000000000001451572323100165365ustar00rootroot00000000000000pineapple-pictures-0.7.3/.reuse/dep5000066400000000000000000000013321451572323100173150ustar00rootroot00000000000000Format: https://www.debian.org/doc/packaging-manuals/copyright-format/1.0/ Upstream-Name: Pineapple Pictures Source: https://github.com/BLumia/pineapple-pictures # Config files Files: .gitignore appveyor.yml .github/* Copyright: None License: CC0-1.0 # README, resource files and Metadata files Files: README*.md assets/*.rc assets/*.qrc dist/* Copyright: None License: CC0-1.0 # Translation files # See assets/plain/translators.html for a list of translators Files: app/translations/*.ts assets/plain/translators.html Copyright: Translators from hosted.weblate.org License: MIT # Assets Files: assets/icons/*.svg Copyright: 2022 Gary Wang License: MIT Files: assets/icons/app-icon.* Copyright: 2020 Lovelyblack License: MIT pineapple-pictures-0.7.3/CMakeLists.txt000066400000000000000000000163511451572323100201030ustar00rootroot00000000000000# SPDX-FileCopyrightText: 2022 - 2023 Gary Wang # # SPDX-License-Identifier: MIT project (pineapple-pictures) cmake_minimum_required (VERSION 3.9.5) list(APPEND CMAKE_MODULE_PATH ${CMAKE_CURRENT_LIST_DIR}/cmake) include (GNUInstallDirs) include (FeatureSummary) option (EXIV2_METADATA_SUPPORT "Better image metadata support via libexiv2" ON) option (PREFER_QT_5 "Prefer to use Qt 5 even if we have Qt 6" ON) set (CMAKE_AUTOMOC ON) set (CMAKE_AUTORCC ON) if (PREFER_QT_5) find_package(QT NAMES Qt5 REQUIRED COMPONENTS Core) else () find_package(QT NAMES Qt6 Qt5 REQUIRED COMPONENTS Core) endif () if (${QT_VERSION_MAJOR} EQUAL "5") set (QT_MINIMUM_VERSION "5.15.2") else () set (QT_MINIMUM_VERSION "6.4") endif () find_package(Qt${QT_VERSION_MAJOR} ${QT_MINIMUM_VERSION} REQUIRED COMPONENTS Widgets Svg LinguistTools OPTIONAL_COMPONENTS DBus ) if (${QT_VERSION_MAJOR} EQUAL "6") find_package(Qt${QT_DEFAULT_MAJOR_VERSION} ${QT_MINIMUM_VERSION} CONFIG REQUIRED SvgWidgets) endif () 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}) qt_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} Qt${QT_VERSION_MAJOR}::Widgets Qt${QT_VERSION_MAJOR}::Svg) if (${QT_VERSION_MAJOR} EQUAL "6") target_link_libraries (${EXE_NAME} Qt::SvgWidgets) endif () 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 () if (Qt5DBus_FOUND OR Qt6DBus_FOUND) target_link_libraries (${EXE_NAME} Qt${QT_VERSION_MAJOR}::DBus ) target_compile_definitions(${EXE_NAME} PRIVATE HAVE_QTDBUS ) 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 () # Install settings if (WIN32) # TODO: try to avoid install to a "bin" subfolder under windows... # when fixed, don't forget to update the CI config file... elseif (UNIX) if (CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT) set(CMAKE_INSTALL_PREFIX /usr) endif () # install icon install ( FILES assets/icons/app-icon.svg DESTINATION "${CMAKE_INSTALL_DATADIR}/icons/hicolor/scalable/apps" RENAME net.blumia.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 ${CMAKE_INSTALL_BINDIR} LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR} ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR} 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.7.3/LICENSE000066400000000000000000000020741451572323100163450ustar00rootroot00000000000000MIT 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.7.3/LICENSES/000077500000000000000000000000001451572323100165425ustar00rootroot00000000000000pineapple-pictures-0.7.3/LICENSES/BSD-3-Clause.txt000066400000000000000000000026641451572323100212750ustar00rootroot00000000000000Copyright (c) . Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. 3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. pineapple-pictures-0.7.3/LICENSES/CC0-1.0.txt000066400000000000000000000156101451572323100201470ustar00rootroot00000000000000Creative Commons Legal Code CC0 1.0 Universal CREATIVE COMMONS CORPORATION IS NOT A LAW FIRM AND DOES NOT PROVIDE LEGAL SERVICES. DISTRIBUTION OF THIS DOCUMENT DOES NOT CREATE AN ATTORNEY-CLIENT RELATIONSHIP. CREATIVE COMMONS PROVIDES THIS INFORMATION ON AN "AS-IS" BASIS. CREATIVE COMMONS MAKES NO WARRANTIES REGARDING THE USE OF THIS DOCUMENT OR THE INFORMATION OR WORKS PROVIDED HEREUNDER, AND DISCLAIMS LIABILITY FOR DAMAGES RESULTING FROM THE USE OF THIS DOCUMENT OR THE INFORMATION OR WORKS PROVIDED HEREUNDER. Statement of Purpose The laws of most jurisdictions throughout the world automatically confer exclusive Copyright and Related Rights (defined below) upon the creator and subsequent owner(s) (each and all, an "owner") of an original work of authorship and/or a database (each, a "Work"). Certain owners wish to permanently relinquish those rights to a Work for the purpose of contributing to a commons of creative, cultural and scientific works ("Commons") that the public can reliably and without fear of later claims of infringement build upon, modify, incorporate in other works, reuse and redistribute as freely as possible in any form whatsoever and for any purposes, including without limitation commercial purposes. These owners may contribute to the Commons to promote the ideal of a free culture and the further production of creative, cultural and scientific works, or to gain reputation or greater distribution for their Work in part through the use and efforts of others. For these and/or other purposes and motivations, and without any expectation of additional consideration or compensation, the person associating CC0 with a Work (the "Affirmer"), to the extent that he or she is an owner of Copyright and Related Rights in the Work, voluntarily elects to apply CC0 to the Work and publicly distribute the Work under its terms, with knowledge of his or her Copyright and Related Rights in the Work and the meaning and intended legal effect of CC0 on those rights. 1. Copyright and Related Rights. A Work made available under CC0 may be protected by copyright and related or neighboring rights ("Copyright and Related Rights"). Copyright and Related Rights include, but are not limited to, the following: i. the right to reproduce, adapt, distribute, perform, display, communicate, and translate a Work; ii. moral rights retained by the original author(s) and/or performer(s); iii. publicity and privacy rights pertaining to a person's image or likeness depicted in a Work; iv. rights protecting against unfair competition in regards to a Work, subject to the limitations in paragraph 4(a), below; v. rights protecting the extraction, dissemination, use and reuse of data in a Work; vi. database rights (such as those arising under Directive 96/9/EC of the European Parliament and of the Council of 11 March 1996 on the legal protection of databases, and under any national implementation thereof, including any amended or successor version of such directive); and vii. other similar, equivalent or corresponding rights throughout the world based on applicable law or treaty, and any national implementations thereof. 2. Waiver. To the greatest extent permitted by, but not in contravention of, applicable law, Affirmer hereby overtly, fully, permanently, irrevocably and unconditionally waives, abandons, and surrenders all of Affirmer's Copyright and Related Rights and associated claims and causes of action, whether now known or unknown (including existing as well as future claims and causes of action), in the Work (i) in all territories worldwide, (ii) for the maximum duration provided by applicable law or treaty (including future time extensions), (iii) in any current or future medium and for any number of copies, and (iv) for any purpose whatsoever, including without limitation commercial, advertising or promotional purposes (the "Waiver"). Affirmer makes the Waiver for the benefit of each member of the public at large and to the detriment of Affirmer's heirs and successors, fully intending that such Waiver shall not be subject to revocation, rescission, cancellation, termination, or any other legal or equitable action to disrupt the quiet enjoyment of the Work by the public as contemplated by Affirmer's express Statement of Purpose. 3. Public License Fallback. Should any part of the Waiver for any reason be judged legally invalid or ineffective under applicable law, then the Waiver shall be preserved to the maximum extent permitted taking into account Affirmer's express Statement of Purpose. In addition, to the extent the Waiver is so judged Affirmer hereby grants to each affected person a royalty-free, non transferable, non sublicensable, non exclusive, irrevocable and unconditional license to exercise Affirmer's Copyright and Related Rights in the Work (i) in all territories worldwide, (ii) for the maximum duration provided by applicable law or treaty (including future time extensions), (iii) in any current or future medium and for any number of copies, and (iv) for any purpose whatsoever, including without limitation commercial, advertising or promotional purposes (the "License"). The License shall be deemed effective as of the date CC0 was applied by Affirmer to the Work. Should any part of the License for any reason be judged legally invalid or ineffective under applicable law, such partial invalidity or ineffectiveness shall not invalidate the remainder of the License, and in such case Affirmer hereby affirms that he or she will not (i) exercise any of his or her remaining Copyright and Related Rights in the Work or (ii) assert any associated claims and causes of action with respect to the Work, in either case contrary to Affirmer's express Statement of Purpose. 4. Limitations and Disclaimers. a. No trademark or patent rights held by Affirmer are waived, abandoned, surrendered, licensed or otherwise affected by this document. b. Affirmer offers the Work as-is and makes no representations or warranties of any kind concerning the Work, express, implied, statutory or otherwise, including without limitation warranties of title, merchantability, fitness for a particular purpose, non infringement, or the absence of latent or other defects, accuracy, or the present or absence of errors, whether or not discoverable, all to the greatest extent permissible under applicable law. c. Affirmer disclaims responsibility for clearing rights of other persons that may apply to the Work or any use thereof, including without limitation any person's Copyright and Related Rights in the Work. Further, Affirmer disclaims responsibility for obtaining any necessary consents, permissions or other rights required for any use of the Work. d. Affirmer understands and acknowledges that Creative Commons is not a party to this document and has no duty or obligation with respect to this CC0 or use of the Work. pineapple-pictures-0.7.3/LICENSES/MIT.txt000066400000000000000000000020661451572323100177400ustar00rootroot00000000000000MIT License Copyright (c) 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.7.3/NEWS000066400000000000000000000020451451572323100160350ustar00rootroot00000000000000Version 0.7.3 ~~~~~~~~~~~~~ Released: 2023-10-24 Features: * Add "Keep transformation" to menu Contributors: mmahhi, VenusGirl, albanobattistella, gallegonovato, Heimen Stoffels Version 0.7.2 ~~~~~~~~~~~~~ Released: 2023-08-27 Features: * Add an option in setting dialog to tweak the High-DPI scaling rounding policy (might only works in Qt 6 build) Bugfixes: * Remove image size limit for Qt 6 build * Fix application icon install location under Linux Contributors: Heimen Stoffels, Andrey, Dan, gallegonovato, albanobattistella, Sabri Ünal Version 0.7.1 ~~~~~~~~~~~~~ Released: 2023-07-08 Features: * TIF and TIFF format files in the same folder will now be automatedly added to the gallery * Built-in window resizing now also supports Linux desktop. (macOS might also works as well) Bugfixes: * Settings dialog will automatedly use a suitable size instead of a hard-coded one * Fix default configuration file location under Linux. (was `~/.config/config.ini`, now it's `~/.config/Pineapple Pictures/config.ini`) Contributors: yyc12345 pineapple-pictures-0.7.3/README.md000066400000000000000000000124321451572323100166160ustar00rootroot00000000000000Yet 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/actions/workflows/macos.yml/badge.svg)](https://github.com/BLumia/pineapple-pictures/actions/workflows/macos.yml)| |Ubuntu Build|[![Ubuntu CI](https://github.com/BLumia/pineapple-pictures/actions/workflows/ubuntu.yml/badge.svg)](https://github.com/BLumia/pineapple-pictures/actions/workflows/ubuntu.yml)| ![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. > **Note** > Although there is a `pineapple-pictures.pro` file which can be used for QMake build, it's only for testing purpose and it doesn't have `exiv2` support included. Using QMake to build this project is NOT supported, please use CMake if possible. ### 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.7.3/README.zh_CN.md000066400000000000000000000135021451572323100176150ustar00rootroot00000000000000简单轻量的跨平台看图工具。 |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/actions/workflows/macos.yml/badge.svg)](https://github.com/BLumia/pineapple-pictures/actions/workflows/macos.yml)| |Ubuntu Build|[![Ubuntu CI](https://github.com/BLumia/pineapple-pictures/actions/workflows/ubuntu.yml/badge.svg)](https://github.com/BLumia/pineapple-pictures/actions/workflows/ubuntu.yml)| ![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 用户,您可能需要手动构建和使用图像格式插件。下方给出了进一步的说明。 > **Note** > 尽管存在一个可用于 QMake 构建的 `pineapple-pictures.pro` 文件,但其仅供简单测试所用且其并不包含 `exiv2` 支持。使用 QMake 构建此项目是 **不受支持** 的,请尽可能考虑使用 CMake。 ### 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.7.3/app/000077500000000000000000000000001451572323100161155ustar00rootroot00000000000000pineapple-pictures-0.7.3/app/aboutdialog.cpp000066400000000000000000000203451451572323100211170ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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("
  • %1:
    %2
  • ") .arg(QCoreApplication::translate("MainWindow", "Keep transformation", "The 'transformation' means the flip/rotation status that currently applied to the image view")) .arg(tr("Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images.")), 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) %1 %2", "%1 is year, %2 is the name of copyright holder(s)") .arg(QStringLiteral("2023"), 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) 2023 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.7.3/app/aboutdialog.h000066400000000000000000000015241451572323100205620ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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.7.3/app/actionmanager.cpp000066400000000000000000000150221451572323100214310ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #include "actionmanager.h" #include "mainwindow.h" #include #include #include #define ICON_NAME(name)\ QStringLiteral(":/icons/" #name ".svg") #define ACTION_NAME(s) QStringLiteral(STRIFY(s)) #define STRIFY(s) #s ActionManager::ActionManager() { } ActionManager::~ActionManager() { } QIcon ActionManager::loadHidpiIcon(const QString &resp, QSize sz) { QSvgRenderer r(resp); QPixmap pm = QPixmap(sz * qApp->devicePixelRatio()); pm.fill(Qt::transparent); QPainter p(&pm); r.render(&p); pm.setDevicePixelRatio(qApp->devicePixelRatio()); return QIcon(pm); } void ActionManager::setupAction(MainWindow *mainWindow) { auto create_action = [] (QWidget *w, QAction **a, QString i, QString an) { *a = new QAction(w); if (!i.isNull()) (*a)->setIcon(ActionManager::loadHidpiIcon(i)); (*a)->setObjectName(an); w->addAction(*a); }; #define CREATE_NEW_ICON_ACTION(w, a, i) create_action(w, &a, ICON_NAME(i), ACTION_NAME(a)) 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); #undef CREATE_NEW_ICON_ACTION #define CREATE_NEW_ACTION(w, a) create_action(w, &a, QString(), ACTION_NAME(a)) 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, actionToggleAvoidResetTransform); CREATE_NEW_ACTION(mainWindow, actionSettings); CREATE_NEW_ACTION(mainWindow, actionHelp); CREATE_NEW_ACTION(mainWindow, actionLocateInFileManager); CREATE_NEW_ACTION(mainWindow, actionProperties); CREATE_NEW_ACTION(mainWindow, actionQuitApp); #undef CREATE_NEW_ACTION 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)); actionToggleAvoidResetTransform->setText(QCoreApplication::translate("MainWindow", "Keep transformation", "The 'transformation' means the flip/rotation status that currently applied to the image view")); actionSettings->setText(QCoreApplication::translate("MainWindow", "Configure...", nullptr)); actionHelp->setText(QCoreApplication::translate("MainWindow", "Help", nullptr)); #ifdef Q_OS_WIN actionLocateInFileManager->setText( QCoreApplication::translate( "MainWindow", "Show in File Explorer", "File Explorer is the name of explorer.exe under Windows" ) ); #else actionLocateInFileManager->setText(QCoreApplication::translate("MainWindow", "Show in directory", nullptr)); #endif // Q_OS_WIN 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.7.3/app/actionmanager.h000066400000000000000000000024321451572323100210770ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #ifndef ACTIONMANAGER_H #define ACTIONMANAGER_H #include class MainWindow; class ActionManager { public: ActionManager(); ~ActionManager(); void setupAction(MainWindow * mainWindow); void retranslateUi(MainWindow *MainWindow); void setupShortcuts(); static QIcon loadHidpiIcon(const QString &resp, QSize sz = QSize(32, 32)); 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 *actionToggleAvoidResetTransform; QAction *actionSettings; QAction *actionHelp; QAction *actionLocateInFileManager; QAction *actionProperties; QAction *actionQuitApp; }; #endif // ACTIONMANAGER_H pineapple-pictures-0.7.3/app/bottombuttongroup.cpp000066400000000000000000000035201451572323100224360ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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(32, 32)); 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.7.3/app/bottombuttongroup.h000066400000000000000000000011651451572323100221060ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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.7.3/app/exiv2wrapper.cpp000066400000000000000000000076021451572323100212640ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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() { } #ifdef HAVE_EXIV2_VERSION // stupid AppleClang... 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; // We might get exceptions like "No namespace info available for XMP prefix `Item'" // when trying to get tagLabel() data from a Xmpdatum if the tag is not common-used. // We don't care for those rare tags so let's just use a try-cache... try { 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; #if EXIV2_TEST_VERSION(0, 28, 0) } catch (Exiv2::Error & err) { #else // 0.27.x } catch (Exiv2::AnyError & err) { #endif // EXIV2_TEST_VERSION(0, 28, 0) qWarning() << "Error loading key" << key << ":" << err.what(); } } } #endif // HAVE_EXIV2_VERSION 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.7.3/app/exiv2wrapper.h000066400000000000000000000017061451572323100207300ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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.7.3/app/framelesswindow.cpp000066400000000000000000000111041451572323100220270ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // SPDX-FileCopyrightText: 2023 Tad Young // // SPDX-License-Identifier: MIT #include "framelesswindow.h" #include #include #include #include #include FramelessWindow::FramelessWindow(QWidget *parent) : QWidget(parent) , m_centralLayout(new QVBoxLayout(this)) , m_oldCursorShape(Qt::ArrowCursor) , m_oldEdges() { #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) this->setWindowFlags(Qt::Window | Qt::FramelessWindowHint | Qt::WindowMinMaxButtonsHint); #else // There is a bug in Qt 5 that will make pressing Meta+Up cause the app // fullscreen under Windows, see QTBUG-91226 to learn more. // The bug seems no longer exists in Qt 6 (I only tested it under Qt 6.3.0). this->setWindowFlags(Qt::Window | Qt::FramelessWindowHint | Qt::WindowMinimizeButtonHint); #endif // QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) this->setMouseTracking(true); this->setAttribute(Qt::WA_Hover, true); this->installEventFilter(this); 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; } void FramelessWindow::installResizeCapture(QObject* widget) { widget->installEventFilter(this); } bool FramelessWindow::eventFilter(QObject* o, QEvent* e) { switch (e->type()) { case QEvent::HoverMove: { QWidget* wg = qobject_cast(o); if (wg != nullptr) return mouseHover(static_cast(e), wg); break; } case QEvent::MouseButtonPress: return mousePress(static_cast(e)); } return QWidget::eventFilter(o, e); } bool FramelessWindow::mouseHover(QHoverEvent* event, QWidget* wg) { if (!isMaximized() && !isFullScreen()) { QWindow* win = window()->windowHandle(); Qt::Edges edges = this->getEdgesByPos(wg->mapToGlobal(event->oldPos()), win->frameGeometry()); // backup & restore cursor shape if (edges && !m_oldEdges) // entering the edge. backup cursor shape m_oldCursorShape = win->cursor().shape(); if (!edges && m_oldEdges) // leaving the edge. restore cursor shape win->setCursor(m_oldCursorShape); // save the latest edges status m_oldEdges = edges; // show resize cursor shape if cursor is within border if (edges) { win->setCursor(this->getCursorByEdge(edges, Qt::ArrowCursor)); return true; } } return false; } bool FramelessWindow::mousePress(QMouseEvent* event) { if (event->buttons() & Qt::LeftButton && !isMaximized() && !isFullScreen()) { QWindow* win = window()->windowHandle(); #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) Qt::Edges edges = this->getEdgesByPos(event->globalPosition().toPoint(), win->frameGeometry()); #else Qt::Edges edges = this->getEdgesByPos(event->globalPos(), win->frameGeometry()); #endif // QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) if (edges) { win->startSystemResize(edges); return true; } } return false; } Qt::CursorShape FramelessWindow::getCursorByEdge(const Qt::Edges& edges, Qt::CursorShape default_cursor) { if ((edges == (Qt::TopEdge | Qt::LeftEdge)) || (edges == (Qt::RightEdge | Qt::BottomEdge))) return Qt::SizeFDiagCursor; else if ((edges == (Qt::TopEdge | Qt::RightEdge)) || (edges == (Qt::LeftEdge | Qt::BottomEdge))) return Qt::SizeBDiagCursor; else if (edges & (Qt::TopEdge | Qt::BottomEdge)) return Qt::SizeVerCursor; else if (edges & (Qt::LeftEdge | Qt::RightEdge)) return Qt::SizeHorCursor; else return default_cursor; } Qt::Edges FramelessWindow::getEdgesByPos(const QPoint gpos, const QRect& winrect) { const int borderWidth = 8; Qt::Edges edges; int x = gpos.x() - winrect.x(); int y = gpos.y() - winrect.y(); if (x < borderWidth) edges |= Qt::LeftEdge; if (x > (winrect.width() - borderWidth)) edges |= Qt::RightEdge; if (y < borderWidth) edges |= Qt::TopEdge; if (y > (winrect.height() - borderWidth)) edges |= Qt::BottomEdge; return edges; } pineapple-pictures-0.7.3/app/framelesswindow.h000066400000000000000000000020301451572323100214720ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #ifndef FRAMELESSWINDOW_H #define FRAMELESSWINDOW_H #include QT_BEGIN_NAMESPACE class QVBoxLayout; QT_END_NAMESPACE class FramelessWindow : public QWidget { Q_OBJECT public: explicit FramelessWindow(QWidget *parent = nullptr); void setCentralWidget(QWidget * widget); void installResizeCapture(QObject* widget); protected: bool eventFilter(QObject *o, QEvent *e) override; bool mouseHover(QHoverEvent* event, QWidget* wg); bool mousePress(QMouseEvent* event); private: Qt::Edges m_oldEdges; Qt::CursorShape m_oldCursorShape; Qt::CursorShape getCursorByEdge(const Qt::Edges& edges, Qt::CursorShape default_cursor); Qt::Edges getEdgesByPos(const QPoint pos, const QRect& winrect); QVBoxLayout * m_centralLayout = nullptr; QWidget * m_centralWidget = nullptr; // just a pointer, doesn't take the ownership. }; #endif // FRAMELESSWINDOW_H pineapple-pictures-0.7.3/app/graphicsscene.cpp000066400000000000000000000107601451572323100214430ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #include "graphicsscene.h" #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; } const QPixmap & scaledPixmap(float scaleHint) { if (qFuzzyCompare(scaleHint, m_cachedScaleHint)) return m_cachedPixmap; QSizeF resizedScale(boundingRect().size()); resizedScale *= scaleHint; QPixmap && sourcePixmap = pixmap(); m_cachedPixmap = sourcePixmap.scaled( resizedScale.toSize() * sourcePixmap.devicePixelRatioF(), Qt::KeepAspectRatio, Qt::SmoothTransformation); m_cachedScaleHint = scaleHint; return m_cachedPixmap; } 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); painter->drawPixmap(QRectF(offset(), boundingRect().size()).toRect(), scaledPixmap(m_scaleHint)); } } private: float m_scaleHint = 1; float m_cachedScaleHint = -1; QPixmap m_cachedPixmap; }; class PGraphicsMovieItem : public QGraphicsItem { public: PGraphicsMovieItem(QGraphicsItem *parent = nullptr) : QGraphicsItem(parent) {} void setMovie(QMovie* movie) { if (m_movie) m_movie->disconnect(); m_movie.reset(movie); m_movie->connect(m_movie.data(), &QMovie::updated, [this](){ this->update(); }); } QRectF boundingRect() const override { if (m_movie) { return m_movie->frameRect(); } else { return QRectF(); } } void paint(QPainter* painter, const QStyleOptionGraphicsItem* option, QWidget* widget) override { if (m_movie) { painter->drawPixmap(m_movie->frameRect(), m_movie->currentPixmap(), m_movie->frameRect()); } } private: QScopedPointer m_movie; }; 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(); PGraphicsMovieItem * animatedItem = new PGraphicsMovieItem(); QMovie * movie = new QMovie(filepath); movie->start(); animatedItem->setMovie(movie); this->addItem(animatedItem); m_theThing = animatedItem; 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.7.3/app/graphicsscene.h000066400000000000000000000012421451572323100211030ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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.7.3/app/graphicsview.cpp000066400000000000000000000315251451572323100213220ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #include "graphicsview.h" #include "graphicsscene.h" #include #include #include #include #include #include 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); #if QT_VERSION > QT_VERSION_CHECK(6, 0, 0) imageReader.setAllocationLimit(0); #endif //QT_VERSION > QT_VERSION_CHECK(6, 0, 0) // 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 { QPixmap && pixmap = QPixmap::fromImageReader(&imageReader); if (pixmap.isNull()) { doRequestGallery = false; showText(tr("Image data is invalid or currently unsupported")); } else { pixmap.setDevicePixelRatio(devicePixelRatioF()); 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() { if (!m_avoidResetTransform) { 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 (m_avoidResetTransform) { emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform()); return; } 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; } bool GraphicsView::avoidResetTransform() const { return m_avoidResetTransform; } void GraphicsView::setAvoidResetTransform(bool avoidReset) { m_avoidResetTransform = avoidReset; } 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); constexpr QColor color(45, 45, 45, 170); constexpr 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.7.3/app/graphicsview.h000066400000000000000000000054411451572323100207650ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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); bool avoidResetTransform() const; void setAvoidResetTransform(bool avoidReset); 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_avoidResetTransform = false; bool m_checkerboardEnabled = false; bool m_isLastCheckerboardColorInverted = false; }; #endif // GRAPHICSVIEW_H pineapple-pictures-0.7.3/app/main.cpp000066400000000000000000000034261451572323100175520ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #include "mainwindow.h" #include "playlistmanager.h" #include "settings.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[]) { QGuiApplication::setHighDpiScaleFactorRoundingPolicy(Settings::instance()->hiDpiScaleFactorBehavior()); QApplication a(argc, argv); #if QT_VERSION < QT_VERSION_CHECK(6, 0, 0) a.setAttribute(Qt::ApplicationAttribute::AA_UseHighDpiPixmaps); #endif QTranslator translator; QString qmDir; #ifdef _WIN32 qmDir = QDir(QCoreApplication::applicationDirPath()).absoluteFilePath("translations"); #else qmDir = QT_STRINGIFY(QM_FILE_INSTALL_DIR); #endif if (translator.load(QLocale(), QLatin1String("PineapplePictures"), QLatin1String("_"), 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.initWindowSize(); return a.exec(); } pineapple-pictures-0.7.3/app/mainwindow.cpp000066400000000000000000000571701451572323100210070ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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 #include #include #ifdef HAVE_QTDBUS #include #include #endif // HAVE_QTDBUS 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", "*.tif", "*.tiff" }); 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, const 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->setIconSize(QSize(32, 32)); m_closeButton->setFixedSize(QSize(50, 50)); m_closeButton->setIconResourcePath(":/icons/window-close.svg"); connect(m_closeButton, &QAbstractButton::clicked, this, &MainWindow::closeWindow); m_prevButton = new ToolButton(false, m_graphicsView); m_prevButton->setIconSize(QSize(75, 75)); m_prevButton->setIconResourcePath(":/icons/go-previous.svg"); m_prevButton->setVisible(false); m_prevButton->setOpacity(0, false); m_nextButton = new ToolButton(false, m_graphicsView); m_nextButton->setIconSize(QSize(75, 75)); m_nextButton->setIconResourcePath(":/icons/go-next.svg"); 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); }); connect(m_pm, &PlaylistManager::currentIndexChanged, this, [this]() { int index; QUrl url; std::tie(index, url) = m_pm->currentFileUrl(); if (index != -1) { this->setWindowTitle(url.fileName()); } }); QShortcut * fullscreenShorucut = new QShortcut(QKeySequence(QKeySequence::FullScreen), this); connect(fullscreenShorucut, &QShortcut::activated, this, &MainWindow::toggleFullscreen); centerWindow(); QTimer::singleShot(0, this, [this](){ m_am->setupShortcuts(); }); // allow some mouse events can go through these widgets for resizing window. installResizeCapture(m_closeButton); installResizeCapture(m_graphicsView); installResizeCapture(m_graphicsView->viewport()); installResizeCapture(m_gv); installResizeCapture(m_gv->viewport()); } 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::initWindowSize() { switch (Settings::instance()->initWindowSizeBehavior()) { case Settings::WindowSizeBehavior::Auto: adjustWindowSizeBySceneRect(); break; case Settings::WindowSizeBehavior::Maximized: showMaximized(); break; default: adjustWindowSizeBySceneRect(); break; } } void MainWindow::adjustWindowSizeBySceneRect() { if (m_pm->count() < 1) return; 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() && !isFullScreen()) { if (!window()->windowHandle()->startSystemMove()) { #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) move(event->globalPosition().toPoint() - m_oldMousePos); #else move(event->globalPos() - m_oldMousePos); #endif // QT_VERSION >= QT_VERSION_CHECK(6, 0, 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 Settings::DoubleClickBehavior::Close: quitAppAction(); event->accept(); break; case Settings::DoubleClickBehavior::Maximize: toggleMaximize(); event->accept(); break; case Settings::DoubleClickBehavior::Ignore: 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() == Settings::MouseWheelBehavior::Zoom; // 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 * avoidResetTransform = m_am->actionToggleAvoidResetTransform; avoidResetTransform->setCheckable(true); avoidResetTransform->setChecked(m_graphicsView->avoidResetTransform()); 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->addAction(avoidResetTransform); menu->addSeparator(); menu->addAction(toggleSettings); menu->addAction(helpAction); if (currentFileUrl.isValid()) { menu->addSeparator(); if (currentFileUrl.isLocalFile()) { menu->addAction(m_am->actionLocateInFileManager); } 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(); } void MainWindow::toggleAvoidResetTransform() { m_graphicsView->setAvoidResetTransform(!m_graphicsView->avoidResetTransform()); } 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_actionToggleAvoidResetTransform_triggered() { toggleAvoidResetTransform(); } 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_actionLocateInFileManager_triggered() { QUrl currentFileUrl = currentImageFileUrl(); if (!currentFileUrl.isValid()) return; QFileInfo fileInfo(currentFileUrl.toLocalFile()); if (!fileInfo.exists()) return; QUrl && folderUrl = QUrl::fromLocalFile(fileInfo.absolutePath()); #ifdef Q_OS_WIN QProcess::startDetached("explorer", QStringList() << "/select," << QDir::toNativeSeparators(fileInfo.absoluteFilePath())); #elif defined(Q_OS_LINUX) and defined(HAVE_QTDBUS) // Use https://www.freedesktop.org/wiki/Specifications/file-manager-interface/ if possible const QDBusConnectionInterface * dbusIface = QDBusConnection::sessionBus().interface(); if (!dbusIface || !dbusIface->isServiceRegistered(QLatin1String("org.freedesktop.FileManager1"))) { QDesktopServices::openUrl(folderUrl); return; } QDBusInterface fm1Iface(QStringLiteral("org.freedesktop.FileManager1"), QStringLiteral("/org/freedesktop/FileManager1"), QStringLiteral("org.freedesktop.FileManager1")); fm1Iface.setTimeout(1000); fm1Iface.callWithArgumentList(QDBus::Block, "ShowItems", { QStringList{currentFileUrl.toString()}, QString() }); if (fm1Iface.lastError().isValid()) { QDesktopServices::openUrl(folderUrl); } #else QDesktopServices::openUrl(folderUrl); #endif // Q_OS_WIN } void MainWindow::on_actionQuitApp_triggered() { quitAppAction(false); } pineapple-pictures-0.7.3/app/mainwindow.h000066400000000000000000000071271451572323100204510ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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 initWindowSize(); 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(); void toggleAvoidResetTransform(); 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_actionToggleAvoidResetTransform_triggered(); void on_actionSettings_triggered(); void on_actionHelp_triggered(); void on_actionProperties_triggered(); void on_actionLocateInFileManager_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.7.3/app/metadatadialog.cpp000066400000000000000000000056321451572323100215670ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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.7.3/app/metadatadialog.h000066400000000000000000000011131451572323100212220ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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.7.3/app/metadatamodel.cpp000066400000000000000000000360311451572323100214250ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #include "metadatamodel.h" #include "exiv2wrapper.h" #include #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"), QDir::toNativeSeparators(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.SubjectDistance"), tr("Subject distance")); 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.Contrast"), tr("Contrast")); 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.7.3/app/metadatamodel.h000066400000000000000000000042771451572323100211010ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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.7.3/app/navigatorview.cpp000066400000000000000000000037631451572323100215170ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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.7.3/app/navigatorview.h000066400000000000000000000017311451572323100211550ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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.7.3/app/opacityhelper.cpp000066400000000000000000000015071451572323100214740ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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.7.3/app/opacityhelper.h000066400000000000000000000010551451572323100211370ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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.7.3/app/playlistmanager.cpp000066400000000000000000000110331451572323100220130ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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 currentIndexChanged(m_currentIndex); emit loaded(m_playlist.count()); } void PlaylistManager::setCurrentIndex(int index) { if (index < 0 || index >= m_playlist.count()) return; m_currentIndex = index; emit currentIndexChanged(m_currentIndex); } 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); } int PlaylistManager::count() const { return m_playlist.count(); } 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.7.3/app/playlistmanager.h000066400000000000000000000031271451572323100214650ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #pragma once #include class PlaylistManager : public QObject { Q_OBJECT public: Q_PROPERTY(int currentIndex MEMBER m_currentIndex NOTIFY currentIndexChanged) 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); int count() const; 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); void currentIndexChanged(int index); private: QList m_playlist; PlaylistType m_type; QString m_currentDir; int m_currentIndex = -1; QStringList m_autoLoadSuffix = {}; }; pineapple-pictures-0.7.3/app/settings.cpp000066400000000000000000000112761451572323100204700ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #include "settings.h" #include #include #include #include #include namespace QEnumHelper { template E fromString(const QString &text, const E defaultValue) { bool ok; E result = static_cast(QMetaEnum::fromType().keyToValue(text.toUtf8(), &ok)); if (!ok) { return defaultValue; } return result; } template QString toString(E value) { const int intValue = static_cast(value); return QString::fromUtf8(QMetaEnum::fromType().valueToKey(intValue)); } } 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(); } Settings::DoubleClickBehavior Settings::doubleClickBehavior() const { QString result = m_qsettings->value("double_click_behavior", "Close").toString(); return QEnumHelper::fromString(result, DoubleClickBehavior::Close); } Settings::MouseWheelBehavior Settings::mouseWheelBehavior() const { QString result = m_qsettings->value("mouse_wheel_behavior", "Zoom").toString(); return QEnumHelper::fromString(result, MouseWheelBehavior::Zoom); } Settings::WindowSizeBehavior Settings::initWindowSizeBehavior() const { QString result = m_qsettings->value("init_window_size_behavior", "Auto").toString(); return QEnumHelper::fromString(result, WindowSizeBehavior::Auto); } Qt::HighDpiScaleFactorRoundingPolicy Settings::hiDpiScaleFactorBehavior() const { QString result = m_qsettings->value("hidpi_scale_factor_behavior", "PassThrough").toString(); return QEnumHelper::fromString(result, Qt::HighDpiScaleFactorRoundingPolicy::PassThrough); } 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", QEnumHelper::toString(dcb)); m_qsettings->sync(); } void Settings::setMouseWheelBehavior(MouseWheelBehavior mwb) { m_qsettings->setValue("mouse_wheel_behavior", QEnumHelper::toString(mwb)); m_qsettings->sync(); } void Settings::setInitWindowSizeBehavior(WindowSizeBehavior wsb) { m_qsettings->setValue("init_window_size_behavior", QEnumHelper::toString(wsb)); m_qsettings->sync(); } void Settings::setHiDpiScaleFactorBehavior(Qt::HighDpiScaleFactorRoundingPolicy hidpi) { m_qsettings->setValue("hidpi_scale_factor_behavior", QEnumHelper::toString(hidpi)); m_qsettings->sync(); } #if defined(FLAG_PORTABLE_MODE_SUPPORT) && defined(Q_OS_WIN) #include // QCoreApplication::applicationDirPath() parses the "applicationDirPath" from arg0, which... // 1. rely on a QApplication object instance // but we need to call QGuiApplication::setHighDpiScaleFactorRoundingPolicy() before QApplication get created // 2. arg0 is NOT garanteed to be the path of execution // see also: https://stackoverflow.com/questions/383973/is-args0-guaranteed-to-be-the-path-of-execution // This function is here mainly for #1. QString getApplicationDirPath() { WCHAR buffer[MAX_PATH]; GetModuleFileNameW(NULL, buffer, MAX_PATH); QString appPath = QString::fromWCharArray(buffer); return appPath.left(appPath.lastIndexOf('\\')); } #endif // defined(FLAG_PORTABLE_MODE_SUPPORT) && defined(Q_OS_WIN) Settings::Settings() : QObject(qApp) { QString configPath; #if defined(FLAG_PORTABLE_MODE_SUPPORT) && defined(Q_OS_WIN) QString portableConfigDirPath = QDir(getApplicationDirPath()).absoluteFilePath("data"); QFileInfo portableConfigDirInfo(portableConfigDirPath); if (portableConfigDirInfo.exists() && portableConfigDirInfo.isDir() && portableConfigDirInfo.isWritable()) { // we can use it. configPath = portableConfigDirPath; } #endif // defined(FLAG_PORTABLE_MODE_SUPPORT) && defined(Q_OS_WIN) if (configPath.isEmpty()) { // %LOCALAPPDATA%\ under Windows, ~/.config/ under Linux. configPath = QStandardPaths::writableLocation(QStandardPaths::AppConfigLocation); } m_qsettings = new QSettings(QDir(configPath).absoluteFilePath("config.ini"), QSettings::IniFormat, this); } pineapple-pictures-0.7.3/app/settings.h000066400000000000000000000024221451572323100201260ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #pragma once #include #include class Settings : public QObject { Q_OBJECT public: enum DoubleClickBehavior { Ignore, Close, Maximize, }; Q_ENUM(DoubleClickBehavior) enum MouseWheelBehavior { Zoom, Switch, }; Q_ENUM(MouseWheelBehavior) enum WindowSizeBehavior { Auto, Maximized, }; Q_ENUM(WindowSizeBehavior) static Settings *instance(); bool stayOnTop(); DoubleClickBehavior doubleClickBehavior() const; MouseWheelBehavior mouseWheelBehavior() const; WindowSizeBehavior initWindowSizeBehavior() const; Qt::HighDpiScaleFactorRoundingPolicy hiDpiScaleFactorBehavior() const; void setStayOnTop(bool on); void setDoubleClickBehavior(DoubleClickBehavior dcb); void setMouseWheelBehavior(MouseWheelBehavior mwb); void setInitWindowSizeBehavior(WindowSizeBehavior wsb); void setHiDpiScaleFactorBehavior(Qt::HighDpiScaleFactorRoundingPolicy hidpi); private: Settings(); static Settings *m_settings_instance; QSettings *m_qsettings; signals: public slots: }; pineapple-pictures-0.7.3/app/settingsdialog.cpp000066400000000000000000000125611451572323100216460ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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) , m_initWindowSizeBehavior(new QComboBox) , m_hiDpiRoundingPolicyBehavior(new QComboBox) { this->setWindowTitle(tr("Settings")); QFormLayout * settingsForm = new QFormLayout(this); static QList< QPair > _dc_options { { Settings::DoubleClickBehavior::Ignore, tr("Do nothing") }, { Settings::DoubleClickBehavior::Close, tr("Close the window") }, { Settings::DoubleClickBehavior::Maximize, tr("Toggle maximize") } }; static QList< QPair > _mw_options { { Settings::MouseWheelBehavior::Zoom, tr("Zoom in and out") }, { Settings::MouseWheelBehavior::Switch, tr("View next or previous item") } }; static QList< QPair > _iws_options { { Settings::WindowSizeBehavior::Auto, tr("Auto size") }, { Settings::WindowSizeBehavior::Maximized, tr("Maximized") } }; static QList< QPair > _hidpi_options { { Qt::HighDpiScaleFactorRoundingPolicy::Round, tr("Round (Integer scaling)", "This option means round up for .5 and above") }, { Qt::HighDpiScaleFactorRoundingPolicy::Ceil, tr("Ceil (Integer scaling)", "This option means always round up") }, { Qt::HighDpiScaleFactorRoundingPolicy::Floor, tr("Floor (Integer scaling)", "This option means always round down") }, { Qt::HighDpiScaleFactorRoundingPolicy::PassThrough, tr("Follow system (Fractional scaling)", "This option means don't round") } }; QStringList dcbDropDown; for (const QPair & dcOption : _dc_options) { dcbDropDown.append(dcOption.second); } QStringList mwbDropDown; for (const QPair & mwOption : _mw_options) { mwbDropDown.append(mwOption.second); } QStringList iwsbDropDown; for (const QPair & iwsOption : _iws_options) { iwsbDropDown.append(iwsOption.second); } QStringList hidpiDropDown; for (const QPair & hidpiOption : _hidpi_options) { hidpiDropDown.append(hidpiOption.second); } 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); settingsForm->addRow(tr("Default window size"), m_initWindowSizeBehavior); settingsForm->addRow(tr("HiDPI scale factor rounding policy"), m_hiDpiRoundingPolicyBehavior); m_stayOnTop->setChecked(Settings::instance()->stayOnTop()); m_doubleClickBehavior->setModel(new QStringListModel(dcbDropDown)); Settings::DoubleClickBehavior dcb = Settings::instance()->doubleClickBehavior(); m_doubleClickBehavior->setCurrentIndex(static_cast(dcb)); m_mouseWheelBehavior->setModel(new QStringListModel(mwbDropDown)); Settings::MouseWheelBehavior mwb = Settings::instance()->mouseWheelBehavior(); m_mouseWheelBehavior->setCurrentIndex(static_cast(mwb)); m_initWindowSizeBehavior->setModel(new QStringListModel(iwsbDropDown)); Settings::WindowSizeBehavior iwsb = Settings::instance()->initWindowSizeBehavior(); m_initWindowSizeBehavior->setCurrentIndex(static_cast(iwsb)); m_hiDpiRoundingPolicyBehavior->setModel(new QStringListModel(hidpiDropDown)); Qt::HighDpiScaleFactorRoundingPolicy hidpi = Settings::instance()->hiDpiScaleFactorBehavior(); for (int i = 0; i < _hidpi_options.count(); i++) { if (_hidpi_options.at(i).first == hidpi) { m_hiDpiRoundingPolicyBehavior->setCurrentIndex(i); break; } } 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(_dc_options.at(index).first); }); connect(m_mouseWheelBehavior, QOverload::of(&QComboBox::currentIndexChanged), this, [ = ](int index){ Settings::instance()->setMouseWheelBehavior(_mw_options.at(index).first); }); connect(m_initWindowSizeBehavior, QOverload::of(&QComboBox::currentIndexChanged), this, [ = ](int index){ Settings::instance()->setInitWindowSizeBehavior(_iws_options.at(index).first); }); connect(m_hiDpiRoundingPolicyBehavior, QOverload::of(&QComboBox::currentIndexChanged), this, [ = ](int index){ Settings::instance()->setHiDpiScaleFactorBehavior(_hidpi_options.at(index).first); }); adjustSize(); setWindowFlag(Qt::WindowContextHelpButtonHint, false); } SettingsDialog::~SettingsDialog() { } pineapple-pictures-0.7.3/app/settingsdialog.h000066400000000000000000000013061451572323100213060ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #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; QComboBox * m_initWindowSizeBehavior = nullptr; QComboBox * m_hiDpiRoundingPolicyBehavior = nullptr; }; #endif // SETTINGSDIALOG_H pineapple-pictures-0.7.3/app/toolbutton.cpp000066400000000000000000000016751451572323100210430ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #include "toolbutton.h" #include "actionmanager.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::setIconResourcePath(const QString &iconp) { this->setIcon(ActionManager::loadHidpiIcon(iconp, this->iconSize())); } void ToolButton::setOpacity(qreal opacity, bool animated) { m_opacityHelper->setOpacity(opacity, animated); } pineapple-pictures-0.7.3/app/toolbutton.h000066400000000000000000000010411451572323100204730ustar00rootroot00000000000000// SPDX-FileCopyrightText: 2022 Gary Wang // // SPDX-License-Identifier: MIT #ifndef TOOLBUTTON_H #define TOOLBUTTON_H #include class OpacityHelper; class ToolButton : public QPushButton { Q_OBJECT public: ToolButton(bool hoverColor = false, QWidget * parent = nullptr); void setIconResourcePath(const QString &iconp); public slots: void setOpacity(qreal opacity, bool animated = true); private: OpacityHelper * m_opacityHelper; }; #endif // TOOLBUTTON_H pineapple-pictures-0.7.3/app/translations/000077500000000000000000000000001451572323100206365ustar00rootroot00000000000000pineapple-pictures-0.7.3/app/translations/PineapplePictures.ts000066400000000000000000000703721451572323100246530ustar00rootroot00000000000000 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) Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Version: %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 Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) %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 Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Zoom in Zoom out Flip &Horizontally &Paste Toggle Checkerboard &Open... Actual size Toggle maximize Rotate right Previous image Next image Configure... Help Show in File Explorer File Explorer is the name of explorer.exe under Windows Show in directory 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 Subject distance Flash mode 35mm focal length Lens model Contrast 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 Auto size Maximized Round (Integer scaling) This option means round up for .5 and above Ceil (Integer scaling) This option means always round up Floor (Integer scaling) This option means always round down Follow system (Fractional scaling) This option means don't round Stay on top when start-up Double-click behavior Mouse wheel behavior Default window size HiDPI scale factor rounding policy main Pineapple Pictures File list. pineapple-pictures-0.7.3/app/translations/PineapplePictures_ca.ts000066400000000000000000000717441451572323100253220ustar00rootroot00000000000000 AboutDialog About Quant a Launch application with image file path as argument to load the file. Inicia l'aplicació amb el camí del fitxer de la imatge com a argument per carregar la imatge. Drag and drop image file onto the window is also supported. També podeu arrossegar i deixar anar un fitxer d'imatge a la finestra. None of the operations in this application will alter the pictures on disk. Cap de les operacions en aquesta aplicació alterarà els fitxers d'imatge. Context menu option explanation: Explicació de les opcions del menú contextual: Make window stay on top of all other windows. Manté la finestra a sobre de totes les altres finestres. Avoid close window accidentally. (eg. by double clicking the window) Evita que es tanqui la finestra accidentalment (com ara en fer doble clic a la finestra) Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Version: %1 Versió: %1 Logo designed by %1 Logotip dissenyat por %1 Built with Qt %1 (%2) Creat amb Qt %1 (%2) Source code Codi font Contributors Col·laboradors List of contributors on GitHub Llista de col·laboradors al GitHub Thanks to all people who contributed to this project. Gràcies a totes les persones que han col·laborat en aquest projecte. Translators Traductors I would like to thank the following people who volunteered to translate this application. M'agradaria donar les gràcies a les persones següents per oferir-se a traduir aquesta aplicació. %1 is built on the following free software libraries: Free as in freedom %1 està construït sobre les biblioteques de programari lliure següents: &Special Thanks &Especial agraïment &Third-party Libraries &Biblioteques de tercers Your Rights Els vostres drets Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) Copyright (c) %1 %2 %1 is released under the MIT License. %1 es publica sota la llicència MIT. This license grants people a number of freedoms: Aquesta llicència atorga a les persones diverses llibertats: You are free to use %1, for any purpose Sou lliure de fer servir %1 per a qualsevol propòsit You are free to distribute %1 Sou lliure de distribuir %1 You can study how %1 works and change it Podeu estudiar com funciona %1 i modificar-lo You can distribute changed versions of %1 Podeu distribuir les versions modificades de %1 The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. La llicència del MIT us garanteix aquesta llibertat. No és permès que ningú us la tregui. Third-party Libraries used by %1 Biblioteques de tercers que fa servir %1 &Help Aj&uda &About &Quant a &License &Llicència GraphicsScene Drag image here Arrossegueu una imatge aquí GraphicsView File url list is empty La llista d'ubicacions de fitxer és buida File is not a valid image El fitxer no és una imatge vàlida Image data is invalid or currently unsupported Les dades de la imatge no són vàlides o no són compatibles Image data is invalid Les dades de la imatge no són vàlides Not supported mimedata: %1 El tipus MIME no és compatible: %1 MainWindow File url list is empty La llista d'ubicacions és buida &Copy &Copia Copy P&ixmap Copia el &mapa de píxels Copy &File Path Copia el camí del &fitxer Properties Propietats Stay on top Mantén a sobre Protected mode Mode protegit Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Zoom in Amplia Zoom out Redueix Flip &Horizontally Inverteix &horitzontalment &Paste &Enganxa Toggle Checkerboard Commuta el tauler d'escacs &Open... &Obre... Actual size Mida real Toggle maximize Commuta la maximització Rotate right Commuta la maximització Previous image Imatge anterior Next image Imatge següent Configure... Configura... Help Ajuda Show in File Explorer File Explorer is the name of explorer.exe under Windows Mostra al navegador de fitxers Show in directory Mostra a la carpeta Quit Surt MetadataDialog Image Metadata Metadades de la imatge MetadataModel Origin Section name. Origen Image Section name. Imatge File Section name. Fitxer Camera Section name. Càmera %1 File Fitxer %1 Description Section name. Descripció Advanced photo Section name. Foto avançada GPS Section name. GPS Dimensions Dimensions Aspect ratio Relació d'aspecte Frame count Núm. d'imatges Name Nom Item type Tipus d'element Folder path Camí de la carpeta Size Mida Date created Data de creació Date modified Data de modificació Title Títol Subject Tema Rating Valoració Tags Etiquetes Comments Comentaris Authors Autors Date taken Data de la foto Program name Nom del programa Copyright Copyright Horizontal resolution Resolució horitzontal Vertical resolution Resolució vertical Resolution unit Unitat de resolució Colour representation Representació del color Camera maker Fabricant de la càmera Camera model Model de la càmera F-stop Relació focal Exposure time Temps d'exposició ISO speed Sensibilitat ISO Exposure bias Compensació d'exposició Focal length Distància focal Max aperture Obertura màxima Metering mode Mode de mesura Subject distance Flash mode Mode del flaix 35mm focal length Distància focal de 35 mm Lens model Model de lent Contrast Brightness Brillantor Exposure program Programa d'exposició Saturation Saturació Sharpness Nitidesa White balance Balanç de blancs Digital zoom Zoom digital EXIF version Versió de l'EXIF Latitude reference Referència de la latitud Latitude Latitud Longitude reference Referència de la longitud Longitude Longitud Altitude reference Referència de l'altitud Altitude Altitud %1 x %2 %1 x %2 %1 : %2 %1 : %2 Property Propietat Value Valor SettingsDialog Settings Paràmetres Do nothing No facis res Close the window Tanca la finestra Toggle maximize Commuta la maximització Zoom in and out Amplia i redueix View next or previous item Mostra l'element següent o l'anterior Auto size Mida automàtica Maximized Maximitza Round (Integer scaling) This option means round up for .5 and above Ceil (Integer scaling) This option means always round up Floor (Integer scaling) This option means always round down Follow system (Fractional scaling) This option means don't round Stay on top when start-up Mantingues a sobre a l'inici Double-click behavior Comportament del doble clic Mouse wheel behavior Comportament de la roda del ratolí Default window size Mida de la finestra per defecte HiDPI scale factor rounding policy main Pineapple Pictures Pineapple Pictures File list. Llista de fitxers. pineapple-pictures-0.7.3/app/translations/PineapplePictures_de.ts000066400000000000000000000715211451572323100253200ustar00rootroot00000000000000 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. Keine der Änderungen in dieser Anwendung modifizieren die abgespeicherten Bilder. 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 Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) Copyright (c) %1 %2 %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 Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Zoom in Hineinzoomen Zoom out Herauszoomen Flip &Horizontally &Horizontal spiegeln &Paste %Einfügen Toggle Checkerboard Schachbrettmuster umschalten &Open... &Öffnen... Actual size Tatsächliche Größe Toggle maximize Maximieren umschalten Rotate right Nach rechts drehen Previous image Vorheriges Bild Next image Nächstes Bild Configure... Konfigurieren … Help Hilfe Show in File Explorer File Explorer is the name of explorer.exe under Windows Im Dateiexplorer zeigen Show in directory Im Verzeichnis zeigen Quit Beenden 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 Framezahl Name Name Item type Objekttyp Folder path Ordnerpfad Size Größe Date created Datum erstellt Date modified Datum geändert Title Titel Subject Betreff Rating Bewertung Tags Tags Comments Kommentare Authors Autoren Date taken Datum genommen Program name Programmname Copyright 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 Subject distance Flash mode Flash-Modus 35mm focal length 35 mm Brennweite Lens model Objektivmodell Contrast 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 Breitengradbezug Latitude Breitengrad Longitude reference Längengradbezug Longitude Längengrad Altitude reference Höhenbezug 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 Hinein- und Hinauszoomen View next or previous item Zeige nächstes oder vorheriges Element Auto size Automatische Größe Maximized Maximiert Round (Integer scaling) This option means round up for .5 and above Ceil (Integer scaling) This option means always round up Floor (Integer scaling) This option means always round down Follow system (Fractional scaling) This option means don't round Stay on top when start-up Beim Start oben bleiben Double-click behavior Doppelklickverhalten Mouse wheel behavior Mausradverhalten Default window size Standard-Fenstergröße HiDPI scale factor rounding policy main Pineapple Pictures Pineapple Pictures File list. Dateiliste. pineapple-pictures-0.7.3/app/translations/PineapplePictures_es.ts000066400000000000000000000724311451572323100253400ustar00rootroot00000000000000 AboutDialog About Acerca de Launch application with image file path as argument to load the file. Inicia la aplicación con la ruta del archivo de la imagen como argumento para cargar la imagen. Drag and drop image file onto the window is also supported. También es posible arrastrar y soltar un archivo de imagen en la ventana. None of the operations in this application will alter the pictures on disk. Ninguna de las operaciones en esta aplicación alterará los archivos de imagen. Context menu option explanation: Explicación de las opciones del menú contextual: Make window stay on top of all other windows. Mantiene la ventana encima de todas las demás ventanas. Avoid close window accidentally. (eg. by double clicking the window) Evita que se cierre la ventana accidentalmente (por ejemplo, al hacer 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) Creado 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 han colaborado en 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 personas siguientes por ofrecerse a traducir esta aplicación. %1 is built on the following free software libraries: Free as in freedom %1 está construido sobre las bibliotecas de software libre siguientes: &Special Thanks Agradecimiento &especial &Third-party Libraries &Bibliotecas de terceros Your Rights Sus derechos Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Evita restablecer el estado de zoom/rotación/inversión que se aplicó a la vista de la imagen al cambiar entre las imágenes. Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) Copyright (c) %1 %2 %1 is released under the MIT License. %1 se publica bajo la licencia MIT. This license grants people a number of freedoms: Esta licencia otorga a las personas una serie de libertades: You are free to use %1, for any purpose Es libre de usar %1 para cualquier propósito You are free to distribute %1 Es libre de distribuir %1 You can study how %1 works and change it Puede estudiar cómo funciona %1 y modificarlo You can distribute changed versions of %1 Puede distribuir versiones modificadas de %1 The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. La licencia MIT le garantiza esta libertad. Nadie está autorizado a quitársela. Third-party Libraries used by %1 Bibliotecas de terceros usadas por %1 &Help Ay&uda &About &Acerca de &License &Licencia GraphicsScene Drag image here Arrastre una imagen aquí GraphicsView File url list is empty La lista de ubicaciones está vacía File is not a valid image El archivo no es una imagen válida Image data is invalid or currently unsupported Los datos de la imagen no son válidos o no son compatibles Image data is invalid Los datos de la imagen no son válidos Not supported mimedata: %1 El tipo MIME no es compatible: %1 MainWindow File url list is empty La lista de ubicaciones está vacía &Copy &Copiar Copy P&ixmap Copiar &mapa de píxeles Copy &File Path Copiar &ruta de archivo Properties Propiedades Stay on top Mantener encima Protected mode Modo protegido Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Conservar la transformación Zoom in Ampliar Zoom out Reducir Flip &Horizontally Voltear &horizontalmente &Paste &Pegar Toggle Checkerboard Activar/desactivar el tablero de ajedrez &Open... &Abrir... Actual size Tamaño real Toggle maximize Maximizar/desmaximizar Rotate right Girar a la derecha Previous image Imagen anterior Next image Imagen siguiente Configure... Configurar... Help Ayuda Show in File Explorer File Explorer is the name of explorer.exe under Windows Mostrar en el Explorador de archivos Show in directory Mostrar en la carpeta Quit Salir MetadataDialog Image Metadata Metadatos de la imagen MetadataModel Origin Section name. Origen Image Section name. Imagen File Section name. Archivo Camera Section name. Cámara %1 File Archivo %1 Description Section name. Descripción Advanced photo Section name. Foto avanzada GPS Section name. GPS Dimensions Dimensiones Aspect ratio Relación de aspecto Frame count Núm. de imágenes Name Nombre Item type Tipo de elemento Folder path Ruta de la carpeta Size Tamaño Date created Fecha de creación Date modified Fecha de modificación Title Título Subject Tema Rating Valoración Tags Etiquetas Comments Comentarios Authors Autores Date taken Fecha en que se tomó Program name Nombre del programa Copyright Copyright Horizontal resolution Resolución horizontal Vertical resolution Resolución vertical Resolution unit Unidad de resolución Colour representation Representación del color Camera maker Fabricante de la cámara Camera model Modelo de la cámara F-stop Relación focal Exposure time Tiempo de exposición ISO speed Sensibilidad ISO Exposure bias Compensación de exposición Focal length Distancia focal Max aperture Apertura máxima Metering mode Modo de medición Subject distance Distancia del sujeto Flash mode Modo del flash 35mm focal length Distancia focal de 35 mm Lens model Modelo de la lente Contrast Contraste Brightness Brillo Exposure program Programa de exposición Saturation Saturación Sharpness Nitidez White balance Balance de blancos Digital zoom Zum digital EXIF version Versión de EXIF Latitude reference Referencia de la latitud Latitude Latitud Longitude reference Referencia de la longitud Longitude Longitud Altitude reference Referencia de la altitud Altitude Altitud %1 x %2 %1 x %2 %1 : %2 %1 : %2 Property Propiedad Value Valor SettingsDialog Settings Preferencias Do nothing No hacer nada Close the window Cerrar la ventana Toggle maximize Maximizar/desmaximizar Zoom in and out Ampliar y reducir View next or previous item Mostrar el elemento siguiente/anterior Auto size Tamaño automático Maximized Maximizar Round (Integer scaling) This option means round up for .5 and above Redondeo (escala de enteros) Ceil (Integer scaling) This option means always round up Ceil (redondear enteros hacia arriba) Floor (Integer scaling) This option means always round down Floor (redondear enteros hacia abajo) Follow system (Fractional scaling) This option means don't round Redondeo (redondear los enteros) Stay on top when start-up Mantener encima al inicio Double-click behavior Comportamiento del doble clic Mouse wheel behavior Comportamiento de la rueda del ratón Default window size Tamaño de la ventana por defecto HiDPI scale factor rounding policy Política de redondeo del factor de escala HiDPI main Pineapple Pictures Pineapple Pictures File list. Lista de archivos. pineapple-pictures-0.7.3/app/translations/PineapplePictures_fr.ts000066400000000000000000000724711451572323100253440ustar00rootroot00000000000000 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 Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) Copyright (c) %1 %2 %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é Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Zoom in Zoom avant Zoom out Zoom arrière Flip &Horizontally Retourner &horizontalement &Paste Co&ller Toggle Checkerboard Dés/activer le damier &Open... &Ouvrir... Actual size Taille actuelle Toggle maximize Dés/activer l'agrandissement Rotate right Pivoter vers la droite Previous image Image précédente Next image Image suivant Configure... Configurer… Help Aide Show in File Explorer File Explorer is the name of explorer.exe under Windows Afficher dans le navigateur de fichiers Show in directory Afficher dans le dossier 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 Subject distance Flash mode Mode flash 35mm focal length Distance focale de 35 mm Lens model Modèle d'objectif Contrast 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 Auto size Taille automatique Maximized Agrandi Round (Integer scaling) This option means round up for .5 and above Ceil (Integer scaling) This option means always round up Floor (Integer scaling) This option means always round down Follow system (Fractional scaling) This option means don't round 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 Default window size Taille de la fenêtre par défaut HiDPI scale factor rounding policy main Pineapple Pictures Pineapple Pictures File list. Liste des fichiers. pineapple-pictures-0.7.3/app/translations/PineapplePictures_id.ts000066400000000000000000000710571451572323100253300ustar00rootroot00000000000000 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 Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) %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 Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Zoom in Perbesar Zoom out Perkecil Flip &Horizontally Putar Secara &Horizontal &Paste &Tempel Toggle Checkerboard &Open... Actual size Ukuran asli Toggle maximize Rotate right Putar ke kanan Previous image Next image Configure... Konfigurasi... Help Dukungan Show in File Explorer File Explorer is the name of explorer.exe under Windows Show in directory 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 Subject distance Flash mode 35mm focal length Lens model Model lensa Contrast 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 Auto size Maximized Round (Integer scaling) This option means round up for .5 and above Ceil (Integer scaling) This option means always round up Floor (Integer scaling) This option means always round down Follow system (Fractional scaling) This option means don't round Stay on top when start-up Double-click behavior Mouse wheel behavior Default window size HiDPI scale factor rounding policy main Pineapple Pictures Pineapple Pictures File list. Daftar berkas. pineapple-pictures-0.7.3/app/translations/PineapplePictures_it.ts000066400000000000000000000722671451572323100253540ustar00rootroot00000000000000 AboutDialog About Informazioni Launch application with image file path as argument to load the file. Avvia l'applicazione con il percorso del file immagine come argomento per caricare il file. Drag and drop image file onto the window is also supported. È supportato anche il trascinamento del file immagine sulla finestra. None of the operations in this application will alter the pictures on disk. Nessuna delle operazioni in questa applicazione altererà le immagini sul disco. Context menu option explanation: Spiegazione delle opzioni del menu contestuale: Make window stay on top of all other windows. Fai in modo che la finestra rimanga in cima a tutte le altre finestre. Avoid close window accidentally. (eg. by double clicking the window) Evitare di chiudere accidentalmente la finestra. (es. facendo doppio clic sulla finestra) Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Evitare di reimpostare lo stato di zoom/rotazione/capovolgimento applicato alla visualizzazione dell'immagine quando si passa da un'immagine all'altra. Version: %1 Versione: %1 Logo designed by %1 Logo disegnato da %1 Built with Qt %1 (%2) Costruito con Qt %1 (%2) Source code Codice sorgente Contributors Contributori List of contributors on GitHub Elenco dei contributori su GitHub Thanks to all people who contributed to this project. Grazie a tutte le persone che hanno contribuito a questo progetto. Translators Traduttori I would like to thank the following people who volunteered to translate this application. Vorrei ringraziare le seguenti persone che si sono offerte volontarie per tradurre questa applicazione. %1 is built on the following free software libraries: Free as in freedom %1 si basa sulle seguenti librerie di software libero: &Special Thanks &Ringraziamenti speciali &Third-party Libraries &Librerie di terze parti Your Rights I tuoi diritti Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) Copyright (c) %1 %2 %1 is released under the MIT License. %1 è rilasciato sotto licenza MIT. This license grants people a number of freedoms: Questa licenza garantisce alle persone una serie di libertà: You are free to use %1, for any purpose Sei libero di usare %1, per qualsiasi scopo You are free to distribute %1 Sei libero di distribuire %1 You can study how %1 works and change it Puoi studiare come funziona %1 e cambiarlo You can distribute changed versions of %1 Puoi distribuire versioni modificate di %1 The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. La licenza MIT ti garantisce questa libertà. A nessuno è mai permesso portarlo via. Third-party Libraries used by %1 Librerie di terze parti utilizzate da %1 &Help &Aiuto &About &Informazioni &License &Licenza GraphicsScene Drag image here Trascina qui l'immagine GraphicsView File url list is empty L'elenco degli URL dei file è vuoto File is not a valid image Il file non è un'immagine valida Image data is invalid or currently unsupported I dati dell'immagine non sono validi o non sono attualmente supportati Image data is invalid I dati dell'immagine non sono validi Not supported mimedata: %1 Dati mime non supportati: %1 MainWindow File url list is empty L'elenco degli URL dei file è vuoto &Copy &Copia Copy P&ixmap Copia P&ixmap Copy &File Path Copia &Percorso file Properties Proprietà Stay on top Rimani in cima Protected mode Modalità protetta Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Mantieni trasformazione Zoom in Zoom avanti Zoom out Zoom indietro Flip &Horizontally Capovolgi &Orizzontalmente &Paste &Incolla Toggle Checkerboard Attiva/disattiva scacchiera &Open... &Apri... Actual size Dimensione reale Toggle maximize Attiva massimizzazione Rotate right Ruota a destra Previous image Immagine precedente Next image Immagine successiva Configure... Configura... Help Aiuto Show in File Explorer File Explorer is the name of explorer.exe under Windows Mostra in Esplora file Show in directory Mostra nella directory Quit Esci MetadataDialog Image Metadata Metadati dell'immagine MetadataModel Origin Section name. Origine Image Section name. Immagine File Section name. File Camera Section name. Camera %1 File %1 File Description Section name. Descrizione Advanced photo Section name. Foto avanzata GPS Section name. GPS Dimensions Dimensioni Aspect ratio Proporzioni Frame count Conteggio fotogrammi Name Nome Item type Tipo di elemento Folder path Percorso cartella Size Dimensione Date created Data di creazione Date modified Data di modifica Title Titolo Subject Soggetto Rating Valutazione Tags Tag Comments Commenti Authors Autori Date taken Data scatto Program name Nome programma Copyright Copyright Horizontal resolution Risoluzione orizzontale Vertical resolution Risoluzione verticale Resolution unit Unità di risoluzione Colour representation Rappresentazione del colore Camera maker Produttore macchina fotografica Camera model Modello camera F-stop F-stop Exposure time Tempo di esposizione ISO speed Velocità ISO Exposure bias Bias d'esposizione Focal length Lunghezza focale Max aperture Massima apertura Metering mode Modalità di misurazione Subject distance Distanza del soggetto Flash mode Modalità flash 35mm focal length Lunghezza focale 35 mm Lens model Modello di lente Contrast Contrasto Brightness Luminosità Exposure program Programma di esposizione Saturation Saturazione Sharpness Nitidezza White balance Bilanciamento del bianco Digital zoom Zoom digitale EXIF version Versione EXIF Latitude reference Riferimento di latitudine Latitude Latitudine Longitude reference Riferimento di longitudine Longitude Longitudine Altitude reference Riferimento altimetrico Altitude Altitudine %1 x %2 %1 x %2 %1 : %2 %1 : %2 Property Proprietà Value Valore SettingsDialog Settings Impostazioni Do nothing Non fare nulla Close the window Chiudi la finestra Toggle maximize Attiva massimizzazione Zoom in and out Zoom avanti e indietro View next or previous item Visualizza l'elemento successivo o precedente Auto size Dimensione automatica Maximized Massimizzato Round (Integer scaling) This option means round up for .5 and above Round (ridimensionamento intero) Ceil (Integer scaling) This option means always round up Ceil (ridimensionamento intero) Floor (Integer scaling) This option means always round down Floor (ridimensionamento intero) Follow system (Fractional scaling) This option means don't round Segui il sistema (scala frazionaria) Stay on top when start-up Rimani in cima quando si avvia Double-click behavior Comportamento del doppio clic Mouse wheel behavior Comportamento della rotellina del mouse Default window size Dimensioni predefinite della finestra HiDPI scale factor rounding policy Politica di arrotondamento del fattore di scala HiDPI main Pineapple Pictures Immagini di Pineapple File list. Elenco file. pineapple-pictures-0.7.3/app/translations/PineapplePictures_ja.ts000066400000000000000000000730511451572323100253220ustar00rootroot00000000000000 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) 誤ってウィンドウを閉じないようにする。(例:ウィンドウをダブルクリックする) Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Version: %1 バージョン: %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 スペシャルサンクス(&S) &Third-party Libraries サードパーティライブラリ(&T) Your Rights 利用者の権利 Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) Copyright (c) %1 %2 %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 プロテクトモード Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Zoom in 拡大 Zoom out 縮小 Flip &Horizontally 画像を左右反転する(&H) &Paste 貼り付け(&P) Toggle Checkerboard 背景を格子模様に切り替え &Open... 開く(&O)… Actual size 実際のサイズ Toggle maximize 最大化を切り替える Rotate right 右に回転 Previous image 前の画像 Next image 次の画像 Configure... 設定... Help ヘルプ Show in File Explorer File Explorer is the name of explorer.exe under Windows エクスプローラーで表示する Show in directory ディレクトリに表示する 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 測光モード Subject distance 被写体距離 Flash mode フラッシュモード 35mm focal length 35mm 焦点距離 Lens model レンズモデル Contrast コントラスト 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 × %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 次の項目または前の項目を表示 Auto size オートサイズ Maximized 最大化 Round (Integer scaling) This option means round up for .5 and above 四捨五入 (整数スケーリング) Ceil (Integer scaling) This option means always round up 切り上げ (整数スケーリング) Floor (Integer scaling) This option means always round down 切り捨て (整数スケーリング) Follow system (Fractional scaling) This option means don't round システム設定に従う (小数スケーリング) Stay on top when start-up 起動時に最前面に表示する Double-click behavior ダブルクリックの動作 Mouse wheel behavior マウスホイールの動作 Default window size 既定のウィンドウサイズ HiDPI scale factor rounding policy 高DPIスケーリングの四捨五入方法 main Pineapple Pictures Pineapple Pictures File list. ファイルリスト pineapple-pictures-0.7.3/app/translations/PineapplePictures_ko.ts000066400000000000000000000716311451572323100253430ustar00rootroot00000000000000 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) 실수로 창을 닫지 마십시오. (예: 창을 두 번 클릭하여) Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Version: %1 버전: %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 특별한 감사(&S) &Third-party Libraries 타사 라이브러리(&T) Your Rights 사용자 권한 Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) 저작권 (c) %1 %2 %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 Pixmap 복사(&I) Copy &File Path 파일 경로 복사(&F) Properties 속성 Stay on top 맨 위에 유지 Protected mode 보호 모드 Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Zoom in 확대 Zoom out 축소 Flip &Horizontally 수평으로 뒤집기(&H) &Paste 붙여넣기(&P) Toggle Checkerboard 바둑판 전환 &Open... 열기(&O)... Actual size 실제 크기 Toggle maximize 최대화 전환 Rotate right 오른쪽으로 회전 Previous image 이전 이미지 Next image 다음 이미지 Configure... 구성... Help 도움말 Show in File Explorer File Explorer is the name of explorer.exe under Windows 파일 탐색기에 표시 Show in directory 디렉터리에 표시 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 F-스톱 Exposure time 노출 시간 ISO speed ISO 속도 Exposure bias 노출 편향 Focal length 초점 거리 Max aperture 최대 조리개 Metering mode 미터링 모드 Subject distance 피사체 거리 Flash mode 플래시 모드 35mm focal length 35mm 초점 거리 Lens model 렌즈 모델 Contrast 대비 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 다음 또는 이전 항목 보기 Auto size 자동 크기 Maximized 최대화 Round (Integer scaling) This option means round up for .5 and above 라운드 (정수 스케일링) Ceil (Integer scaling) This option means always round up 셰일 (정수 스케일링) Floor (Integer scaling) This option means always round down 바닥 (정수 스케일링) Follow system (Fractional scaling) This option means don't round 팔로우 시스템 (부분 스케일링) Stay on top when start-up 시작 시 맨 위에 유지 Double-click behavior 더블클릭 동작 Mouse wheel behavior 마우스 휠 동작 Default window size 기본 창 크기 HiDPI scale factor rounding policy HiDPI 배율 반올림 정책 main Pineapple Pictures 파인애플 픽처스 File list. 파일 목록. pineapple-pictures-0.7.3/app/translations/PineapplePictures_nb_NO.ts000066400000000000000000000712021451572323100257170ustar00rootroot00000000000000 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 Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) Opphavsrett © %1 %2 %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 Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Zoom in Førstørr Zoom out Forminsk Flip &Horizontally Speilvend &horisontalt &Paste &Lim inn Toggle Checkerboard Skru av/på rutemønster &Open... &Åpne … Actual size Faktisk størrelse Toggle maximize Veksle maksimering Rotate right Roter til høyre Previous image Forrige bilde Next image Neste bilde Configure... Sett opp … Help Hjelp Show in File Explorer File Explorer is the name of explorer.exe under Windows Vis i filutforsker Show in directory Vis i mappe 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 Sideforhold 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 Utviklere 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 Subject distance Flash mode Blitz-modus 35mm focal length 35 mm-brennvidde Lens model Linsemodell Contrast Kontrast 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 Auto size Automatisk størrelse Maximized Maksimert Round (Integer scaling) This option means round up for .5 and above Ceil (Integer scaling) This option means always round up Floor (Integer scaling) This option means always round down Follow system (Fractional scaling) This option means don't round Stay on top when start-up Behold i forgrunnen ved oppstart Double-click behavior Dobbeltklikksoppførsel Mouse wheel behavior Musehjulsoppførsel Default window size Forvalgt vindusstørrelse HiDPI scale factor rounding policy main Pineapple Pictures Pineapple Pictures File list. Filliste. pineapple-pictures-0.7.3/app/translations/PineapplePictures_nl.ts000066400000000000000000000714171451572323100253450ustar00rootroot00000000000000 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 Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Voorkom dat het zoomniveau, de draaiing en spiegeling worden hersteld na wisselen van afbeelding. Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) Copyright (c) %1 %2 %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 Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Bewerkingen onthouden Zoom in Inzoomen Zoom out Uitzoomen Flip &Horizontally &Horizontaal spiegelen &Paste &Plakken Toggle Checkerboard Schaakbordpatroon aan/uit &Open... &Openen… Actual size Ware grootte Toggle maximize Maximaliseren aan/uit Rotate right Naar rechts draaien Previous image Vorige afbeelding Next image Volgende afbeelding Configure... Instellen... Help Hulp Show in File Explorer File Explorer is the name of explorer.exe under Windows Tonen in bestandsbeheer Show in directory Tonen in map 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 Subject distance Onderwerpafstand Flash mode Flitsmodus 35mm focal length 35mm focale lengte Lens model Lensmodel Contrast Contrast 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 Auto size Automatische grootte Maximized Gemaximaliseerd Round (Integer scaling) This option means round up for .5 and above Rond (geheel getal) Ceil (Integer scaling) This option means always round up Keil (geheel getal) Floor (Integer scaling) This option means always round down Grond (geheel getal) Follow system (Fractional scaling) This option means don't round Systeeminstelling (fractionele schaal) Stay on top when start-up Automatisch altijd bovenop Double-click behavior Dubbelklikgedrag Mouse wheel behavior Scrollwielgedrag Default window size Standaard vensterafmetingen HiDPI scale factor rounding policy HiDPI-schaalfactor - afrondbeleid main Pineapple Pictures Pineapple Afbeeldingen File list. Bestandslijst. pineapple-pictures-0.7.3/app/translations/PineapplePictures_pa_PK.ts000066400000000000000000000702051451572323100257200ustar00rootroot00000000000000 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) Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Version: %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 Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) %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 سرکھیات سیٹنگ Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Zoom in وڈا کرو Zoom out چھوٹا کرو Flip &Horizontally لیٹویں اُلٹاؤ &Paste پیسٹ کرو Toggle Checkerboard چیک‌بورڈ چالو بدلو &Open... کھُلھو… Actual size اصلی اکار Toggle maximize ودھو ودھ بدلو Rotate right سجے گھنماؤ Previous image پچھلی تصویر Next image اگلی تصویر Configure... Help مدد Show in File Explorer File Explorer is the name of explorer.exe under Windows Show in directory 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 Subject distance Flash mode 35mm focal length Lens model Contrast 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 Auto size Maximized ودھ توں ودھ Round (Integer scaling) This option means round up for .5 and above Ceil (Integer scaling) This option means always round up Floor (Integer scaling) This option means always round down Follow system (Fractional scaling) This option means don't round Stay on top when start-up Double-click behavior Mouse wheel behavior Default window size HiDPI scale factor rounding policy main Pineapple Pictures File list. pineapple-pictures-0.7.3/app/translations/PineapplePictures_ru.ts000066400000000000000000000770511451572323100253620ustar00rootroot00000000000000 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 Ваши Права Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) Copyright (c) %1 %2 %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 Защищенный режим Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Zoom in Увеличить Zoom out Уменьшить Flip &Horizontally Отразить по &горизонтали &Paste &Вставить Toggle Checkerboard Переключить фоновый рисунок &Open... &Открыть... Actual size Фактический размер Toggle maximize Переключить окно Rotate right Повернуть вправо Previous image Предыдущее изображение Next image Следующее изображение Configure... Параметры... Help Помощь Show in File Explorer File Explorer is the name of explorer.exe under Windows Показать в проводнике Show in directory Показать в папке 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 Режим измерения Subject distance Расстояние до объекта Flash mode Режим вспышки 35mm focal length Фокусное расстояние 35 мм Lens model Модель объектива Contrast Контраст 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 Следующее или предыдущее изображение Auto size Авторазмер Maximized Максимизировать Round (Integer scaling) This option means round up for .5 and above Round (целочисленное масштабирование) Ceil (Integer scaling) This option means always round up Ceil (целочисленное масштабирование) Floor (Integer scaling) This option means always round down Floor (целочисленное масштабирование) Follow system (Fractional scaling) This option means don't round Следовать системе (дробное масштабирование) Stay on top when start-up Поверх всех окон при запуске Double-click behavior Действие при двойном щелчке Mouse wheel behavior Действие колеса мыши Default window size Размер окна по умолчанию HiDPI scale factor rounding policy Политика округления коэффициента масштабирования HiDPI main Pineapple Pictures Pineapple Pictures File list. Список файлов. pineapple-pictures-0.7.3/app/translations/PineapplePictures_si.ts000066400000000000000000000737031451572323100253470ustar00rootroot00000000000000 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 ඔබගේ අයිතිවාසිකම් Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) %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 Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Zoom in Zoom out Flip &Horizontally Copy P&ixmap Copy &File Path &Paste Toggle Checkerboard &Open... Actual size Toggle maximize Rotate right Previous image Next image Configure... Help Show in File Explorer File Explorer is the name of explorer.exe under Windows Show in directory 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 Subject distance Flash mode 35mm focal length Lens model කාච ආකෘතිය Contrast 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 Auto size Maximized Round (Integer scaling) This option means round up for .5 and above Ceil (Integer scaling) This option means always round up Floor (Integer scaling) This option means always round down Follow system (Fractional scaling) This option means don't round Stay on top when start-up Double-click behavior Mouse wheel behavior Default window size HiDPI scale factor rounding policy main Pineapple Pictures පයින්ඇපල් පික්චර්ස් File list. ගොනු ලැයිස්තුව. pineapple-pictures-0.7.3/app/translations/PineapplePictures_tr.ts000066400000000000000000000717601451572323100253620ustar00rootroot00000000000000 AboutDialog About Hakkında Launch application with image file path as argument to load the file. Dosyayı yüklemek için değişken olarak resim dosyası yolu ile uygulamayı çalıştır. Drag and drop image file onto the window is also supported. Pencere üzerine resim dosyası sürükle bırak ta destekleniyor. None of the operations in this application will alter the pictures on disk. Bu uygulamadaki işlemlerin hiçbiri diskteki resimleri değiştirmeyecektir. Context menu option explanation: İçerik menüsü seçeneği açıklaması: Make window stay on top of all other windows. Pencereyi diğer tüm pencerelerin üzerinde tut. Avoid close window accidentally. (eg. by double clicking the window) Pencereyi yanlışlıkla kapatmaktan kaçın. (örn. pencereye çift tıklayarak) Version: %1 Sürüm: %1 Copyright (c) 2020 %1 Telif hakkı (c) 2020 %1 Logo designed by %1 Logo tasarımcısı: %1 Built with Qt %1 (%2) Qt %1 (%2) ile inşa edilmiştir Source code Kaynak kodu Contributors Katkıda bulunanlar List of contributors on GitHub Github üzerindeki katkı sağlayıcıların listesi Thanks to all people who contributed to this project. Bu projeye katkı sağlayan herkese teşekkürler. Translators Çevirmenler I would like to thank the following people who volunteered to translate this application. Bu uygulamayı çevirmeye gönüllü olan aşağıdaki kişilere teşekkür etmek istiyorum. %1 is built on the following free software libraries: Free as in freedom %1 aşağıdaki özgür yazılım kitaplıkları üzerinde inşa edilmiştir: &Special Thanks Özel %Teşekkürler &Third-party Libraries Üçüncü Par&ti Kitaplıklar Your Rights Haklarınız Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) Telif Hakkı (c) %1 %2 %1 is released under the MIT License. %1, MIT Lisansı altında sunulmuştur. This license grants people a number of freedoms: Bu lisans birkaç özgürlüğü kişilere veriyor: You are free to use %1, for any purpose %1'i herhangi bir amaç için kullanmakta özgürsünüz You are free to distribute %1 %1'i dağıtmakta özgürsünüz You can study how %1 works and change it %1'in nasıl çalıştığıyla ilgili çalışabilir ve onu değiştirebilirsiniz You can distribute changed versions of %1 %1'in değiştirilmiş sürümünü dağıtabilirsiniz The MIT license guarantees you this freedom. Nobody is ever permitted to take it away. MIT lisansı özgürlüğünüzü garanti eder. Hiç kimsenin özgürlüğünüzü sizden almasına bile izin verilmez. Third-party Libraries used by %1 %1 tarafından kullanılan Üçüncü Parti Kitaplıklar &Help &Yardım &About H&akkında &License &Lisans GraphicsScene Drag image here Resmi buraya sürükleyin GraphicsView File url list is empty Dosya URL listesi boş File is not a valid image Dosya, geçerli bir resim değil Image data is invalid or currently unsupported Resim verisi geçersiz veya şuan desteklenmiyor Image data is invalid Resim verisi geçersiz Not supported mimedata: %1 Desteklenmeyen dosya türü verisi: %1 MainWindow File url list is empty Dosya URL listesi boş &Copy &Kopyala Copy P&ixmap P&ixmap'i Kopyala Copy &File Path &Dosya Yolunu Kopyala Properties Özellikler Stay on top Üstte tut Protected mode Korumalı kip Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Zoom in Yaklaştır Zoom out Uzaklaştır Flip &Horizontally &Yatay Çevir &Paste Ya&pıştır Toggle Checkerboard Damalı Ekrana Geç &Open... &Aç... Actual size Gerçek boyut Toggle maximize Tam boyuta geç Rotate right Sağa döndür Previous image Önceki resim Next image Sonraki resim Configure... Yapılandır... Help Yardım Show in File Explorer File Explorer is the name of explorer.exe under Windows Dosya Gezgini'nde Göster Show in directory Dizinde göster Quit Çıkış MetadataDialog Image Metadata Resim Üstverisi MetadataModel Origin Section name. Köken Image Section name. Resim File Section name. Dosya Camera Section name. Kamera %1 File %1 Dosya Description Section name. Açıklama Advanced photo Section name. Gelişmiş foto GPS Section name. GPS Dimensions Boyutlar Aspect ratio En boy oranı Frame count Kare sayısı Name Adı Item type Öge türü Folder path Klasör yolu Size Boyut Date created Oluşturulma tarihi Date modified Değiştirilme tarihi Title Başlık Subject Konu Rating Değerlendirme Tags Etiketler Comments Yorumlar Authors Sanatçılar Date taken Çekilme tarihi Program name Program adı Copyright Telif Hakkı Horizontal resolution Yatay çözünürlük Vertical resolution Dikey çözünürlük Resolution unit Çözünürlük birimi Colour representation Renk sunumu Camera maker Kamera üreticisi Camera model Kamera modeli F-stop Diyafram Exposure time Pozlama süresi ISO speed ISO hızı Exposure bias Pozlama sapması Focal length Odak uzaklığı Max aperture Azami açıklık Metering mode Ölçme kipi Subject distance Konu mesafesi Flash mode Flaş kipi 35mm focal length 35mm odak uzaklığı Lens model Mercek modeli Contrast Karşıtlık Brightness Parlaklık Exposure program Pozlama programı Saturation Doygunluk Sharpness Keskinlik White balance Beyaz dengesi Digital zoom Dijital yakınlaştırma EXIF version EXIF sürümü Latitude reference Enlem kaynağı Latitude Enlem Longitude reference Boylam kaynağı Longitude Boylam Altitude reference Rakım kaynağı Altitude Rakım %1 x %2 %1 x %2 %1 : %2 %1 : %2 Property Özellik Value Değer SettingsDialog Settings Ayarlar Do nothing Hiçbir şey yapma Close the window Pencereyi kapat Toggle maximize Tam boyuta geç Zoom in and out Yaklaştır ve uzaklaştır View next or previous item Sonraki veya önceki ögeyi görüntüle Auto size Otomatik boyut Maximized Tam boyut Round (Integer scaling) This option means round up for .5 and above Yuvarlak (Tamsayı ölçekleme) Ceil (Integer scaling) This option means always round up Tavan (Tamsayı ölçekleme) Floor (Integer scaling) This option means always round down Kat (Tamsayı ölçekleme) Follow system (Fractional scaling) This option means don't round Sistemi takip et (Kesirli ölçekleme) Stay on top when start-up Açılışta pencerelerin üstünde kal Double-click behavior Çift tıklama davranışı Mouse wheel behavior Fare tekeri davranışı Default window size Öntanımlı pencere boyutu HiDPI scale factor rounding policy HiDPI ölçek katsayısı yuvarlama ilkesi main Pineapple Pictures Ananas Resimler File list. Dosya list. pineapple-pictures-0.7.3/app/translations/PineapplePictures_uk.ts000066400000000000000000000765461451572323100253630ustar00rootroot00000000000000 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) Уникайте випадкового закриття вікна. (наприклад, подвійним клацанням вікна) Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. Version: %1 Версія: %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 Ваші права Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) Авторське право (c) %1 %2 %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 Захищений режим Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view Zoom in Збільшити Zoom out Зменшити Flip &Horizontally Перевернути &Горизонтально &Paste &Вставити Toggle Checkerboard Перемкнути шахову дошку &Open... &Відкрити... Actual size Фактичний розмір Toggle maximize Перемкнути на максимум Rotate right Повернути праворуч Previous image Попереднє зображення Next image Наступне зображення Configure... Налаштувати... Help Допомога Show in File Explorer File Explorer is the name of explorer.exe under Windows Показати у Файловому провіднику Show in directory Показати в каталозі 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 Діафрагма (F) Exposure time Час експозиції ISO speed Чутливість ISO Exposure bias Зсув експозиції Focal length Фокусна відстань Max aperture Максимальна апертура Metering mode Режим вимірювання Subject distance Відстань до об'єкта зйомки Flash mode Режим спалаху 35mm focal length Фокусна відстань 35 мм Lens model Модель об'єктива Contrast Контраст 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 Переглянути наступний або попередній елемент Auto size Автоматичний розмір Maximized Максимізувати Round (Integer scaling) This option means round up for .5 and above Round (цілочисельне масштабування) Ceil (Integer scaling) This option means always round up Ceil (цілочисельне масштабування) Floor (Integer scaling) This option means always round down Floor (цілочисельне масштабування) Follow system (Fractional scaling) This option means don't round Стежити за системою (дробове масштабування) Stay on top when start-up Поверх всіх вікон під час запуску Double-click behavior Поведінка при подвійному кліку Mouse wheel behavior Поведінка колеса миші Default window size Розмір вікна за замовчуванням HiDPI scale factor rounding policy Політика округлення коефіцієнта HiDPI main Pineapple Pictures Pineapple Pictures File list. Список файлів. pineapple-pictures-0.7.3/app/translations/PineapplePictures_zh_CN.ts000066400000000000000000000711121451572323100257250ustar00rootroot00000000000000 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 用户的权利 Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images. 切换图片时,防止重置当前视图的缩放/旋转/翻转状态。 Copyright (c) %1 %2 %1 is year, %2 is the name of copyright holder(s) 版权所有 © %1 %2 %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 保护模式 Keep transformation The 'transformation' means the flip/rotation status that currently applied to the image view 保持视图变换 Zoom in 放大 Zoom out 缩小 Flip &Horizontally 水平翻转(&H) &Paste 粘贴(&P) Toggle Checkerboard 切换棋盘格 &Open... 打开(&O)... Actual size 实际大小 Toggle maximize 最大化窗口 Rotate right 向右旋转 Previous image 上一个图像 Next image 下一个图像 Configure... 设置... Help 帮助 Show in File Explorer File Explorer is the name of explorer.exe under Windows 在文件资源管理器中显示 Show in directory 在文件夹中显示 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 测光模式 Subject distance 目标距离 Flash mode 闪光灯模式 35mm focal length 换算至 35mm 焦距 Lens model 镜头型号 Contrast 对比度 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 查看下一项或上一项 Auto size 自动大小 Maximized 最大化 Round (Integer scaling) This option means round up for .5 and above 四舍五入(整数缩放) Ceil (Integer scaling) This option means always round up 向上取整(整数缩放) Floor (Integer scaling) This option means always round down 向下取整(整数缩放) Follow system (Fractional scaling) This option means don't round 跟随系统(小数缩放) Stay on top when start-up 启动时保持窗口总在最前 Double-click behavior 双击时的行为 Mouse wheel behavior 鼠标滚轮行为 Default window size 默认窗口大小 HiDPI scale factor rounding policy HiDPI 高分屏缩放策略 main Pineapple Pictures 菠萝看图 File list. 文件列表。 pineapple-pictures-0.7.3/appveyor.yml000066400000000000000000000155211451572323100177310ustar00rootroot00000000000000image: - Visual Studio 2022 environment: CMAKE_INSTALL_PREFIX: C:\projects\cmake LIBZ: C:\projects\zlib LIBEXPAT: C:\projects\libexpat LIBAVIF: C:\projects\libavif LIBEXIV2: C:\projects\exiv2 PPKG: C:\projects\ppkg matrix: - job_name: mingw_64_qt6_5 QTDIR: C:\Qt\6.5\mingw_64 MINGW64: C:\Qt\Tools\mingw1120_64 KF_BRANCH: master EXIV2_VERSION: "0.28.0" EXIV2_CMAKE_OPTIONS: "-DEXIV2_ENABLE_BROTLI=OFF -DEXIV2_ENABLE_INIH=OFF -DEXIV2_BUILD_EXIV2_COMMAND=OFF" PPIC_CMAKE_OPTIONS: "-DPREFER_QT_5=OFF" WINDEPLOYQT_ARGS: "--verbose=2 --no-quick-import --no-translations --no-opengl-sw --no-system-d3d-compiler" - job_name: mingw81_64_qt5_15_2 QTDIR: C:\Qt\5.15.2\mingw81_64 MINGW64: C:\Qt\Tools\mingw810_64 KF_BRANCH: kf5 EXIV2_VERSION: "0.27.7" EXIV2_CMAKE_OPTIONS: "-DEXIV2_BUILD_SAMPLES=OFF -DEXIV2_ENABLE_WIN_UNICODE=ON -DEXIV2_BUILD_EXIV2_COMMAND=OFF" PPIC_CMAKE_OPTIONS: "" WINDEPLOYQT_ARGS: "--verbose=2 --no-quick-import --no-translations --no-opengl-sw --no-angle --no-system-d3d-compiler" install: - mkdir %CMAKE_INSTALL_PREFIX% - mkdir %LIBZ% - mkdir %LIBEXPAT% - mkdir %LIBAVIF% - mkdir %LIBEXIV2% - mkdir %PPKG% - cd %APPVEYOR_BUILD_FOLDER% - git submodule update --init --recursive - set PATH=%PATH%;%CMAKE_INSTALL_PREFIX%;%QTDIR%\bin;%MINGW64%\bin;%PPKG% - set CC=%MINGW64%\bin\gcc.exe - set CXX=%MINGW64%\bin\g++.exe build_script: # prepare - mkdir 3rdparty - cinst ninja - cd %PPKG% - curl -fsSL -o ppkg.exe https://github.com/BLumia/pineapple-package-manager/releases/latest/download/ppkg.exe - cd %APPVEYOR_BUILD_FOLDER% # download and install zlib for KArchive - cd %LIBZ% - curl -fsSL -o zlib13.zip https://zlib.net/zlib13.zip - 7z x zlib13.zip -y - cd zlib-1.3 - mkdir build - cd build - cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_PREFIX% - 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 -b %KF_BRANCH% -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_PREFIX% -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.6.0 --depth 1 https://aomedia.googlesource.com/aom #- cd aom #- mkdir build.aom #- cd build.aom #- cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_PREFIX% -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 - mkdir aom - cd aom - curl -fsSL -o ppkg-aom.zip https://sourceforge.net/projects/pineapple-package-manager/files/packages/mingw-w64-x86_64-windows/aom-3.6.0-1.zip - ppkg ppkg-aom.zip - 7z x ppkg-aom.zip LICENSE -y - cd %APPVEYOR_BUILD_FOLDER% # install libavif for avif format support of KImageFormats - cd %LIBAVIF% - curl -fsSL -o libavif-v0_11_1.zip https://github.com/AOMediaCodec/libavif/archive/v0.11.1.zip - 7z x libavif-v0_11_1.zip -y - cd libavif-0.11.1 - mkdir build - cd build - cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_PREFIX% -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 -b %KF_BRANCH% -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_PREFIX% - 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_5_0.zip https://github.com/libexpat/libexpat/archive/R_2_5_0.zip - 7z x R_2_5_0.zip -y - cd libexpat-R_2_5_0/expat/ - cmake -G "Ninja" . -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_PREFIX% -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 v%EXIV2_VERSION%.zip https://github.com/Exiv2/exiv2/archive/v%EXIV2_VERSION%.zip - 7z x v%EXIV2_VERSION%.zip -y - cd exiv2-%EXIV2_VERSION% - cmake -G "Ninja" . -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_PREFIX% %EXIV2_CMAKE_OPTIONS% - cmake --build . --target install/strip - cd %APPVEYOR_BUILD_FOLDER% # install KImageFormats - cd 3rdparty - git clone -b %KF_BRANCH% -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=%QTDIR%\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" %PPIC_CMAKE_OPTIONS% -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_PREFIX%\bin\libaom.dll - copy %CMAKE_INSTALL_PREFIX%\bin\libexpat-1.dll - copy %CMAKE_INSTALL_PREFIX%\bin\libexiv2.dll - copy %CMAKE_INSTALL_PREFIX%\bin\libavif.dll - copy %CMAKE_INSTALL_PREFIX%\bin\libzlib.dll - copy %CMAKE_INSTALL_PREFIX%\bin\libKF?Archive.dll - windeployqt %WINDEPLOYQT_ARGS% .\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_5_0\expat\COPYING License.expat.txt - copy %LIBAVIF%\libavif-0.11.1\LICENSE License.libavif.txt - copy %LIBEXIV2%\exiv2-%EXIV2_VERSION%\COPYING License.exiv2.txt # TODO: Qt, zlib - cd .. # for debug.. - tree /f - cd %APPVEYOR_BUILD_FOLDER% - xcopy %CMAKE_INSTALL_PREFIX% .\cmake-prefix-copy /E /H /C /I artifacts: - path: build\bin - path: cmake-prefix-copy pineapple-pictures-0.7.3/assets/000077500000000000000000000000001451572323100166375ustar00rootroot00000000000000pineapple-pictures-0.7.3/assets/icons/000077500000000000000000000000001451572323100177525ustar00rootroot00000000000000pineapple-pictures-0.7.3/assets/icons/app-icon.ico000066400000000000000000006474121451572323100221720ustar00rootroot00000000000000 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.7.3/assets/icons/app-icon.svg000066400000000000000000001307471451572323100222150ustar00rootroot00000000000000 image/svg+xml pineapple-pictures-0.7.3/assets/icons/go-next.svg000066400000000000000000000020541451572323100220550ustar00rootroot00000000000000 pineapple-pictures-0.7.3/assets/icons/go-previous.svg000066400000000000000000000020561451572323100227550ustar00rootroot00000000000000 pineapple-pictures-0.7.3/assets/icons/object-rotate-right.svg000066400000000000000000000023311451572323100243470ustar00rootroot00000000000000 pineapple-pictures-0.7.3/assets/icons/view-background-checkerboard.svg000066400000000000000000000033551451572323100262020ustar00rootroot00000000000000 pineapple-pictures-0.7.3/assets/icons/view-fullscreen.svg000066400000000000000000000031051451572323100236040ustar00rootroot00000000000000 pineapple-pictures-0.7.3/assets/icons/window-close.svg000066400000000000000000000044411451572323100231100ustar00rootroot00000000000000 image/svg+xml pineapple-pictures-0.7.3/assets/icons/zoom-in.svg000066400000000000000000000021601451572323100220620ustar00rootroot00000000000000 pineapple-pictures-0.7.3/assets/icons/zoom-original.svg000066400000000000000000000027531451572323100232700ustar00rootroot00000000000000 pineapple-pictures-0.7.3/assets/icons/zoom-out.svg000066400000000000000000000022021451572323100222600ustar00rootroot00000000000000 pineapple-pictures-0.7.3/assets/pineapple-pictures.rc000066400000000000000000000010031451572323100227700ustar00rootroot00000000000000IDI_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.7.3/assets/plain/000077500000000000000000000000001451572323100177425ustar00rootroot00000000000000pineapple-pictures-0.7.3/assets/plain/translators.html000066400000000000000000000014301451572323100232020ustar00rootroot00000000000000
  • Catalan: Toni Estévez
  • Chinese (Simplified): Percy Hong
  • Dutch: Heimen Stoffels
  • French: J. Lavoie, K. Herbert, Maxime Leroy
  • German: K. Herbert, J. Lavoie, sal0max
  • Indonesian: liimee
  • Italian: albanobattistella
  • Japanese: Black Cat, mmahhi, Percy Hong
  • Korean: VenusGirl
  • Norwegian Bokmål: Allan Nordhøy, ovl-1
  • Punjabi (Pakistan): bgo-eiu
  • Russian: Sergey Shornikov, Artem, Andrey
  • Sinhala: HelaBasa
  • Spanish: Toni Estévez, Génesis Toxical, William(ѕ)ⁿ, gallegonovato
  • Turkish: E-Akcaer, Oğuz Ersen, Sabri Ünal
  • Ukrainian: Dan
pineapple-pictures-0.7.3/assets/resources.qrc000066400000000000000000000010641451572323100213610ustar00rootroot00000000000000 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.7.3/cmake/000077500000000000000000000000001451572323100164155ustar00rootroot00000000000000pineapple-pictures-0.7.3/cmake/FindLibExiv2.cmake000066400000000000000000000072101451572323100216440ustar00rootroot00000000000000#.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.7.3/dist/000077500000000000000000000000001451572323100163005ustar00rootroot00000000000000pineapple-pictures-0.7.3/dist/appstream/000077500000000000000000000000001451572323100202745ustar00rootroot00000000000000pineapple-pictures-0.7.3/dist/appstream/net.blumia.pineapple-pictures.metainfo.xml000066400000000000000000000124001451572323100304620ustar00rootroot00000000000000 net.blumia.pineapple-pictures Pineapple Pictures Pineapple Afbeeldingen Pineapple Pictures 菠萝看图 Image Viewer Afbeeldingsweergave Просмотр изображений 图像查看器 CC0-1.0 MIT Gary (BLumia) Wang et al. Gary (BLumia) Wang e.a. Gary (BLumia) Wang et al. Gary (BLumia) Wang 等人

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.

Pineapple Afbeeldingen is een licht en eenvoudig te gebruiken afbeeldingsweergaveprogramma met miniatuurnavigatie na inzoomen. Het programma heeft echter geen fotobeheermogelijkheid.

Pineapple Pictures - это легкий и простой в использовании просмотрщик изображений, оснащенный удобной навигацией по миниатюрам при увеличении масштаба и не содержащий никакой поддержки управления изображениями.

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

net.blumia.pineapple-pictures.desktop https://github.com/BLumia/pineapple-pictures https://github.com/BLumia/pineapple-pictures/issues https://hosted.weblate.org/projects/pineapple-pictures/ ppic Main window when an image file is loaded Hoofdvenster na het laden van een afbeelding Основное окно после загрузки файла изображения 加载图片后的主窗口 https://pineapple-pictures.sourceforge.io/ppic-gui-static.png Zooming in a raster image Inzoomen op een roosterafbeelding Масштабирование растрового изображения 放大浏览位图 https://pineapple-pictures.sourceforge.io/ppic-zoom-raster.png Zooming in a vector image Inzoomen op een vectorafbeelding Масштабирование векторного изображения 放大浏览矢量图 https://pineapple-pictures.sourceforge.io/ppic-zoom-svg.png

This release adds the following feature:

  • Add "Keep transformation" to menu

With contributions from:

mmahhi, VenusGirl, albanobattistella, gallegonovato, Heimen Stoffels

This release adds the following feature:

  • Add an option in setting dialog to tweak the High-DPI scaling rounding policy

This release fixes the following bugs:

  • Remove image size limit for Qt 6 build
  • Fix application icon install location under Linux

This release adds the following features:

  • TIF and TIFF format files in the same folder will now be automatedly added to the gallery
  • Built-in window resizing now also supports Linux desktop.

This release fixes the following bugs:

  • Settings dialog will automatedly use a suitable size instead of a hard-coded one
  • Fix default configuration file location under Linux.
pineapple-pictures-0.7.3/dist/appstream/po/000077500000000000000000000000001451572323100207125ustar00rootroot00000000000000pineapple-pictures-0.7.3/dist/appstream/po/net.blumia.pineapple-pictures.metainfo.nl.po000066400000000000000000000036701451572323100313370ustar00rootroot00000000000000msgid "" msgstr "" "Project-Id-Version: PACKAGE VERSION\n" "POT-Creation-Date: 2023-08-22 18:49中国标准时间\n" "PO-Revision-Date: 2023-08-23 06:41+0000\n" "Last-Translator: Heimen Stoffels \n" "Language-Team: Dutch \n" "Language: nl\n" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=UTF-8\n" "Content-Transfer-Encoding: 8bit\n" "Plural-Forms: nplurals=2; plural=n != 1;\n" "X-Generator: Weblate 5.0-dev\n" #. (itstool) path: component/name #: net.blumia.pineapple-pictures.metainfo.xml:7 msgid "Pineapple Pictures" msgstr "Pineapple Afbeeldingen" #. (itstool) path: component/summary #: net.blumia.pineapple-pictures.metainfo.xml:9 msgid "Image Viewer" msgstr "Afbeeldingsweergave" #. (itstool) path: description/p #: net.blumia.pineapple-pictures.metainfo.xml:12 msgid "" "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." msgstr "" "Pineapple Afbeeldingen is een licht en eenvoudig te gebruiken " "afbeeldingsweergaveprogramma met miniatuurnavigatie na inzoomen. Het " "programma heeft echter geen fotobeheermogelijkheid." #. (itstool) path: screenshot/caption #: net.blumia.pineapple-pictures.metainfo.xml:17 msgid "Main window when an image file is loaded" msgstr "Hoofdvenster na het laden van een afbeelding" #. (itstool) path: screenshot/caption #: net.blumia.pineapple-pictures.metainfo.xml:22 msgid "Zooming in a raster image" msgstr "Inzoomen op een roosterafbeelding" #. (itstool) path: screenshot/caption #: net.blumia.pineapple-pictures.metainfo.xml:27 msgid "Zooming in a vector image" msgstr "Inzoomen op een vectorafbeelding" #. (itstool) path: component/developer_name #: net.blumia.pineapple-pictures.metainfo.xml:34 msgid "Gary (BLumia) Wang et al." msgstr "Gary (BLumia) Wang e.a." pineapple-pictures-0.7.3/dist/appstream/po/net.blumia.pineapple-pictures.metainfo.pot000066400000000000000000000026621451572323100311130ustar00rootroot00000000000000msgid "" msgstr "" "Project-Id-Version: PACKAGE VERSION\n" "POT-Creation-Date: 2023-08-22 18:49中国标准时间\n" "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" "Last-Translator: FULL NAME \n" "Language-Team: LANGUAGE \n" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=UTF-8\n" "Content-Transfer-Encoding: 8bit\n" #. (itstool) path: component/name #: net.blumia.pineapple-pictures.metainfo.xml:7 msgid "Pineapple Pictures" msgstr "" #. (itstool) path: component/summary #: net.blumia.pineapple-pictures.metainfo.xml:9 msgid "Image Viewer" msgstr "" #. (itstool) path: description/p #: net.blumia.pineapple-pictures.metainfo.xml:12 msgid "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." msgstr "" #. (itstool) path: screenshot/caption #: net.blumia.pineapple-pictures.metainfo.xml:17 msgid "Main window when an image file is loaded" msgstr "" #. (itstool) path: screenshot/caption #: net.blumia.pineapple-pictures.metainfo.xml:22 msgid "Zooming in a raster image" msgstr "" #. (itstool) path: screenshot/caption #: net.blumia.pineapple-pictures.metainfo.xml:27 msgid "Zooming in a vector image" msgstr "" #. (itstool) path: component/developer_name #: net.blumia.pineapple-pictures.metainfo.xml:34 msgid "Gary (BLumia) Wang et al." msgstr "" pineapple-pictures-0.7.3/dist/appstream/po/net.blumia.pineapple-pictures.metainfo.ru.po000066400000000000000000000045131451572323100313510ustar00rootroot00000000000000msgid "" msgstr "" "Project-Id-Version: PACKAGE VERSION\n" "POT-Creation-Date: 2023-08-22 18:49中国标准时间\n" "PO-Revision-Date: 2023-08-23 06:41+0000\n" "Last-Translator: Andrey \n" "Language-Team: Russian \n" "Language: ru\n" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=UTF-8\n" "Content-Transfer-Encoding: 8bit\n" "Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n" "%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;\n" "X-Generator: Weblate 5.0-dev\n" #. (itstool) path: component/name #: net.blumia.pineapple-pictures.metainfo.xml:7 msgid "Pineapple Pictures" msgstr "Pineapple Pictures" #. (itstool) path: component/summary #: net.blumia.pineapple-pictures.metainfo.xml:9 msgid "Image Viewer" msgstr "Просмотр изображений" #. (itstool) path: description/p #: net.blumia.pineapple-pictures.metainfo.xml:12 msgid "" "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." msgstr "" "Pineapple Pictures - это легкий и простой в использовании просмотрщик " "изображений, оснащенный удобной навигацией по миниатюрам при увеличении " "масштаба и не содержащий никакой поддержки управления изображениями." #. (itstool) path: screenshot/caption #: net.blumia.pineapple-pictures.metainfo.xml:17 msgid "Main window when an image file is loaded" msgstr "Основное окно после загрузки файла изображения" #. (itstool) path: screenshot/caption #: net.blumia.pineapple-pictures.metainfo.xml:22 msgid "Zooming in a raster image" msgstr "Масштабирование растрового изображения" #. (itstool) path: screenshot/caption #: net.blumia.pineapple-pictures.metainfo.xml:27 msgid "Zooming in a vector image" msgstr "Масштабирование векторного изображения" #. (itstool) path: component/developer_name #: net.blumia.pineapple-pictures.metainfo.xml:34 msgid "Gary (BLumia) Wang et al." msgstr "Gary (BLumia) Wang et al." pineapple-pictures-0.7.3/dist/appstream/po/net.blumia.pineapple-pictures.metainfo.zh_CN.po000066400000000000000000000032271451572323100317250ustar00rootroot00000000000000msgid "" msgstr "" "Project-Id-Version: \n" "POT-Creation-Date: 2023-08-22 18:49中国标准时间\n" "PO-Revision-Date: 2023-08-22 18:22+0800\n" "Last-Translator: \n" "Language-Team: \n" "Language: zh_CN\n" "MIME-Version: 1.0\n" "Content-Type: text/plain; charset=UTF-8\n" "Content-Transfer-Encoding: 8bit\n" "X-Generator: Poedit 3.3.2\n" #. (itstool) path: component/name #: net.blumia.pineapple-pictures.metainfo.xml:7 msgid "Pineapple Pictures" msgstr "菠萝看图" #. (itstool) path: component/summary #: net.blumia.pineapple-pictures.metainfo.xml:9 msgid "Image Viewer" msgstr "图像查看器" #. (itstool) path: description/p #: net.blumia.pineapple-pictures.metainfo.xml:12 msgid "" "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." msgstr "" "菠萝看图是一个轻量级易用的图像查看器,在图片放大时提供了方便的鸟瞰导航功能," "且不包含任何图片管理功能。" #. (itstool) path: screenshot/caption #: net.blumia.pineapple-pictures.metainfo.xml:17 msgid "Main window when an image file is loaded" msgstr "加载图片后的主窗口" #. (itstool) path: screenshot/caption #: net.blumia.pineapple-pictures.metainfo.xml:22 msgid "Zooming in a raster image" msgstr "放大浏览位图" #. (itstool) path: screenshot/caption #: net.blumia.pineapple-pictures.metainfo.xml:27 msgid "Zooming in a vector image" msgstr "放大浏览矢量图" #. (itstool) path: component/developer_name #: net.blumia.pineapple-pictures.metainfo.xml:34 msgid "Gary (BLumia) Wang et al." msgstr "Gary (BLumia) Wang 等人" pineapple-pictures-0.7.3/dist/net.blumia.pineapple-pictures.desktop000066400000000000000000000013401451572323100255370ustar00rootroot00000000000000[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=net.blumia.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.7.3/pineapple-pictures.pro000066400000000000000000000060761451572323100217010ustar00rootroot00000000000000# SPDX-FileCopyrightText: 2022 Gary Wang # # SPDX-License-Identifier: MIT QT += 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 # MSVC only, since QMake doesn't have a CMAKE_CXX_STANDARD_LIBRARIES or cpp_winlibs similar thing win32-msvc* { LIBS += -luser32 }