1 Commits

Author SHA1 Message Date
8397cee0c5 ci(windows): additional image formats 2023-09-29 14:10:40 +08:00
83 changed files with 6154 additions and 10953 deletions

View File

@ -1,3 +0,0 @@
# .git-blame-ignore-revs
# CR LF to LF
ed5a6023326fd2ab420ded76976501be33e0b389

5
.gitattributes vendored
View File

@ -1,5 +0,0 @@
*.txt text eol=lf
*.cpp text eol=lf
*.h text eol=lf
*.ui text eol=lf
*.qml text eol=lf

2
.github/FUNDING.yml vendored
View File

@ -1,2 +0,0 @@
ko_fi: blumia
custom: ["https://blumia.itch.io/pineapple-pictures"]

View File

@ -1,6 +1,6 @@
name: macOS CI name: macOS CI
on: [push, pull_request, workflow_dispatch] on: [push, pull_request]
jobs: jobs:
build: build:
@ -8,31 +8,12 @@ jobs:
runs-on: macos-latest runs-on: macos-latest
steps: steps:
- uses: actions/checkout@v4 - uses: actions/checkout@v3
- name: Install Qt - name: Install Qt
uses: jurplel/install-qt-action@v4 uses: jurplel/install-qt-action@v3
with: with:
version: '6.9.1' version: '5.15.2'
modules: 'qtimageformats' - name: Run a qt project
- name: Install Conan and Dependencies
id: conan
working-directory: ./
shell: bash
run: | run: |
pip3 install wheel setuptools cmake ./
pip3 install conan --upgrade make
conan --version
conan profile detect
conan install --requires=exiv2/0.28.3 --generator CMakeDeps --generator CMakeToolchain --build=missing
- name: Build
run: |
cmake . -DTRANSLATION_RESOURCE_EMBEDDING=ON --preset conan-release
cmake --build --preset conan-release
- name: Deploy
run: |
macdeployqt ./ppic.app -dmg
ls
- uses: actions/upload-artifact@v4
with:
name: "macos-bundle"
path: "*.dmg"

View File

@ -7,11 +7,9 @@ name: REUSE Compliance Check
on: [push, pull_request] on: [push, pull_request]
jobs: jobs:
reuse-compliance-check: test:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- name: Checkout - uses: actions/checkout@v2
uses: actions/checkout@v4
- name: REUSE Compliance Check - name: REUSE Compliance Check
uses: fsfe/reuse-action@v5 uses: fsfe/reuse-action@v1

View File

@ -3,14 +3,16 @@ name: Ubuntu CI
on: [push, pull_request] on: [push, pull_request]
jobs: jobs:
ubuntu-24-04-build: ubuntu-22-04-build:
runs-on: ubuntu-24.04
runs-on: ubuntu-22.04
steps: steps:
- uses: actions/checkout@v4 - uses: actions/checkout@v3
- name: Get build dept. - name: Get build dept.
run: | run: |
sudo apt update sudo apt update
sudo apt install cmake qt6-base-dev qt6-svg-dev qt6-tools-dev libexiv2-dev sudo apt install cmake qtbase5-dev libqt5svg5-dev qttools5-dev libexiv2-dev
- name: Build it - name: Build it
run: | run: |
mkdir build mkdir build
@ -22,7 +24,7 @@ jobs:
run: | run: |
cd build cd build
sudo apt install ./*.deb sudo apt install ./*.deb
- uses: actions/upload-artifact@v4 - uses: actions/upload-artifact@v3
with: with:
name: ubuntu-24.04-deb-package name: ubuntu-22.04-deb-package
path: build/*.deb path: build/*.deb

View File

@ -1,28 +1,24 @@
name: Windows CI name: Windows CI
on: [push, pull_request, workflow_dispatch] on: [push, pull_request]
jobs: jobs:
msvc-qmake-build: msvc-build:
strategy: strategy:
matrix: matrix:
include: vs: ['2019']
- qt_ver: '6.9.1' msvc_arch: ['x64']
vs: '2022'
aqt_arch: 'win64_msvc2022_64'
msvc_arch: 'x64'
runs-on: windows-2022 runs-on: windows-2019
steps: steps:
- uses: actions/checkout@v4 - uses: actions/checkout@v3
- name: Install Qt - name: Install Qt
uses: jurplel/install-qt-action@v4 uses: jurplel/install-qt-action@v3
with: with:
arch: ${{ matrix.aqt_arch }} arch: 'win64_msvc2019_64'
version: ${{ matrix.qt_ver }} version: '5.15.2'
modules: 'qtimageformats'
- name: Build - name: Build
shell: cmd shell: cmd
run: | run: |
@ -32,102 +28,3 @@ jobs:
call %VCVARS% ${{ matrix.msvc_arch }} call %VCVARS% ${{ matrix.msvc_arch }}
qmake pineapple-pictures.pro qmake pineapple-pictures.pro
nmake nmake
nmake clean
windeployqt --verbose=2 --no-quick-import --no-translations --no-opengl-sw --no-system-d3d-compiler --no-system-dxc-compiler --skip-plugin-types tls,networkinformation release\ppic.exe
- uses: actions/upload-artifact@v4
with:
name: "windows-msvc${{ matrix.vs }}-qt${{ matrix.qt_ver }}-qmake-package"
path: release/*
msvc-cmake-build:
strategy:
matrix:
include:
- qt_ver: '6.9.1'
vs: '2022'
aqt_arch: 'win64_msvc2022_64'
msvc_arch: 'x64'
runs-on: windows-2022
steps:
- uses: actions/checkout@v4
- name: Install Qt
uses: jurplel/install-qt-action@v4
with:
arch: ${{ matrix.aqt_arch }}
version: ${{ matrix.qt_ver }}
modules: 'qtimageformats'
- name: Build
shell: cmd
run: |
:: ------ env ------
set PWD=%cd%
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 }}
:: ------ dep ------
set CMAKE_PREFIX_PATH=%PWD%/dependencies_bin
mkdir dependencies_src
echo ::group::===== exiv2 =====
curl -fsSL -o exiv2_bin.zip https://github.com/Exiv2/exiv2/releases/download/v0.28.5/exiv2-0.28.5-2022msvc-AMD64.zip
7z x exiv2_bin.zip -y
ren .\exiv2-0.28.5-2022msvc-AMD64 dependencies_bin
echo ::endgroup::
echo ::group::===== zlib =====
curl -fsSL -o zlib_src.zip https://zlib.net/zlib131.zip
7z x zlib_src.zip -y -o"dependencies_src"
ren .\dependencies_src\zlib-1.3.1 zlib || goto :error
cmake ./dependencies_src/zlib -Bbuild_dependencies/zlib -DCMAKE_INSTALL_PREFIX="dependencies_bin" || goto :error
cmake --build build_dependencies/zlib --config Release --target=install || goto :error
curl -fsSL -o expat_src.zip https://github.com/libexpat/libexpat/archive/R_2_6_2.zip
echo ::endgroup::
echo ::group::===== AOM for libavif AVI decoding support =====
git clone -q -b v3.12.0 --depth 1 https://aomedia.googlesource.com/aom dependencies_src/aom
cmake ./dependencies_src/aom -Bbuild_dependencies/aom -DCMAKE_INSTALL_PREFIX="dependencies_bin" -DENABLE_DOCS=OFF -DBUILD_SHARED_LIBS=ON -DAOM_TARGET_CPU=generic -DENABLE_TESTS=OFF -DENABLE_TESTDATA=OFF -DENABLE_TOOLS=OFF -DENABLE_EXAMPLES=0 || goto :error
cmake --build build_dependencies/aom --config Release --target=install || goto :error
echo ::endgroup::
echo ::group::===== libavif =====
curl -fsSL -o libavif-v1_2_1.zip https://github.com/AOMediaCodec/libavif/archive/v1.2.1.zip
7z x libavif-v1_2_1.zip -y -o"dependencies_src"
ren .\dependencies_src\libavif-1.2.1 libavif || goto :error
cmake ./dependencies_src/libavif -Bbuild_dependencies/libavif -DCMAKE_INSTALL_PREFIX="dependencies_bin" -DAVIF_CODEC_AOM=ON -DAVIF_LIBYUV=LOCAL
cmake --build build_dependencies/libavif --config Release --target=install || goto :error
echo ::endgroup::
echo ::group::===== expat =====
7z x expat_src.zip -y -o"dependencies_src"
ren .\dependencies_src\libexpat-R_2_6_2 expat || goto :error
cmake ./dependencies_src/expat/expat -Bbuild_dependencies/expat -DCMAKE_INSTALL_PREFIX="dependencies_bin" || goto :error
cmake --build build_dependencies/expat --config Release --target=install || goto :error
echo ::endgroup::
echo ::group::===== ECM =====
git clone -q https://invent.kde.org/frameworks/extra-cmake-modules.git dependencies_src/extra-cmake-modules
cmake .\dependencies_src\extra-cmake-modules -Bbuild_dependencies/extra-cmake-modules -DCMAKE_INSTALL_PREFIX="dependencies_bin" -DBUILD_TESTING=OFF || goto :error
cmake --build build_dependencies/extra-cmake-modules --config Release --target=install || goto :error
echo ::endgroup::
echo ::group::===== KArchive =====
git clone -q https://invent.kde.org/frameworks/karchive.git dependencies_src/karchive
cmake .\dependencies_src\karchive -Bbuild_dependencies/karchive -DBUILD_TESTING=OFF -DWITH_LIBZSTD=OFF -DWITH_BZIP2=OFF -DWITH_LIBLZMA=OFF -DCMAKE_INSTALL_PREFIX="dependencies_bin" || goto :error
cmake --build build_dependencies/karchive --config Release --target=install || goto :error
echo ::endgroup::
echo ::group::===== KImageFormats =====
git clone -q https://invent.kde.org/frameworks/kimageformats.git dependencies_src/kimageformats
cmake .\dependencies_src\kimageformats -Bbuild_dependencies/kimageformats -DKDE_INSTALL_QTPLUGINDIR=%QT_ROOT_DIR%\plugins || goto :error
cmake --build build_dependencies/kimageformats --config Release --target=install || goto :error
echo ::endgroup::
:: ------ app ------
cmake -Bbuild . -DCMAKE_INSTALL_PREFIX="%PWD%\build\"
cmake --build build --config Release
cmake --build build --config Release --target=install
:: ------ pkg ------
windeployqt --verbose=2 --no-quick-import --no-translations --no-opengl-sw --no-system-d3d-compiler --no-system-dxc-compiler --skip-plugin-types tls,networkinformation build\bin\ppic.exe
robocopy ./dependencies_bin/bin build/bin *.dll
if ErrorLevel 8 (exit /B 1)
copy LICENSE build\bin
exit /B 0
- uses: actions/upload-artifact@v4
with:
name: "windows-msvc${{ matrix.vs }}-qt${{ matrix.qt_ver }}-cmake-package"
path: build/bin/*

5
.gitignore vendored
View File

@ -2,17 +2,12 @@
*.user *.user
*.user.* *.user.*
# Why, macOS, why?
.DS_Store
# Translation files # Translation files
*.qm *.qm
*.mo *.mo
# Generic Build Dir # Generic Build Dir
[Bb]uild/ [Bb]uild/
cmake-build-*/
# IDE/Editor config folders # IDE/Editor config folders
.vscode/ .vscode/
.idea/

28
.reuse/dep5 Normal file
View File

@ -0,0 +1,28 @@
Format: 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

View File

@ -1,41 +1,54 @@
# SPDX-FileCopyrightText: 2022 - 2025 Gary Wang <git@blumia.net> # SPDX-FileCopyrightText: 2022 - 2023 Gary Wang <git@blumia.net>
# #
# SPDX-License-Identifier: MIT # SPDX-License-Identifier: MIT
cmake_minimum_required(VERSION 3.16) project (pineapple-pictures)
project(pineapple-pictures VERSION 1.1.0) # don't forget to update NEWS file and AppStream metadata. cmake_minimum_required (VERSION 3.9.5)
include(GNUInstallDirs) list(APPEND CMAKE_MODULE_PATH ${CMAKE_CURRENT_LIST_DIR}/cmake)
include(FeatureSummary)
include (GNUInstallDirs)
include (FeatureSummary)
option (EXIV2_METADATA_SUPPORT "Better image metadata support via libexiv2" ON) option (EXIV2_METADATA_SUPPORT "Better image metadata support via libexiv2" ON)
option (TRANSLATION_RESOURCE_EMBEDDING "Embedding .qm translation files inside resource" OFF) option (PREFER_QT_5 "Prefer to use Qt 5 even if we have Qt 6" ON)
set (CMAKE_CXX_STANDARD 17)
set (CMAKE_CXX_STANDARD_REQUIRED ON)
set (CMAKE_AUTOMOC ON) set (CMAKE_AUTOMOC ON)
set (CMAKE_AUTORCC ON) set (CMAKE_AUTORCC ON)
find_package(QT NAMES Qt6 REQUIRED COMPONENTS Core) if (PREFER_QT_5)
find_package(QT NAMES Qt5 REQUIRED COMPONENTS Core)
else ()
find_package(QT NAMES Qt6 Qt5 REQUIRED COMPONENTS Core)
endif ()
set (QT_MINIMUM_VERSION "6.4") 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 find_package(Qt${QT_VERSION_MAJOR} ${QT_MINIMUM_VERSION} REQUIRED
COMPONENTS Widgets Svg SvgWidgets LinguistTools COMPONENTS Widgets Svg LinguistTools
OPTIONAL_COMPONENTS DBus 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) if (EXIV2_METADATA_SUPPORT)
find_package(exiv2) find_package(LibExiv2)
set_package_properties(exiv2 PROPERTIES set_package_properties(LibExiv2 PROPERTIES
URL "https://www.exiv2.org" URL "https://www.exiv2.org"
DESCRIPTION "image metadata support" DESCRIPTION "image metadata support"
TYPE RECOMMENDED TYPE OPTIONAL
PURPOSE "Bring better image metadata support" PURPOSE "Bring better image metadata support"
) )
endif () endif ()
#LibExiv2_FOUND
set (PPIC_CPP_FILES set (PPIC_CPP_FILES
app/main.cpp app/main.cpp
app/framelesswindow.cpp app/framelesswindow.cpp
@ -54,8 +67,6 @@ set (PPIC_CPP_FILES
app/metadatadialog.cpp app/metadatadialog.cpp
app/exiv2wrapper.cpp app/exiv2wrapper.cpp
app/playlistmanager.cpp app/playlistmanager.cpp
app/shortcutedit.cpp
app/fileopeneventhandler.cpp
) )
set (PPIC_HEADER_FILES set (PPIC_HEADER_FILES
@ -75,8 +86,6 @@ set (PPIC_HEADER_FILES
app/metadatadialog.h app/metadatadialog.h
app/exiv2wrapper.h app/exiv2wrapper.h
app/playlistmanager.h app/playlistmanager.h
app/shortcutedit.h
app/fileopeneventhandler.h
) )
set (PPIC_QRC_FILES set (PPIC_QRC_FILES
@ -93,6 +102,8 @@ set (EXE_NAME ppic)
file (GLOB PPIC_TS_FILES app/translations/*.ts) file (GLOB PPIC_TS_FILES app/translations/*.ts)
set (PPIC_CPP_FILES_FOR_I18N ${PPIC_CPP_FILES}) 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) if (WIN32)
list(APPEND PPIC_RC_FILES assets/pineapple-pictures.rc) list(APPEND PPIC_RC_FILES assets/pineapple-pictures.rc)
endif () endif ()
@ -102,38 +113,29 @@ add_executable (${EXE_NAME}
${PPIC_CPP_FILES} ${PPIC_CPP_FILES}
${PPIC_QRC_FILES} ${PPIC_QRC_FILES}
${PPIC_RC_FILES} ${PPIC_RC_FILES}
${PPIC_QM_FILES}
) )
set(ADD_TRANSLATIONS_ADDITIONAL_ARGS) 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 (Qt6_VERSION VERSION_GREATER_EQUAL "6.9.0") if (LibExiv2_FOUND)
set(ADD_TRANSLATIONS_ADDITIONAL_ARGS MERGE_QT_TRANSLATIONS) message(INFO ${LibExiv2_INCLUDE_DIRS})
endif() target_include_directories(${EXE_NAME}
PRIVATE
if (TRANSLATION_RESOURCE_EMBEDDING) ${LibExiv2_INCLUDE_DIRS}
qt_add_translations(${EXE_NAME} ${ADD_TRANSLATIONS_ADDITIONAL_ARGS} TS_FILES ${PPIC_TS_FILES}) )
else()
qt_add_translations(${EXE_NAME} ${ADD_TRANSLATIONS_ADDITIONAL_ARGS} TS_FILES ${PPIC_TS_FILES} QM_FILES_OUTPUT_VARIABLE PPIC_QM_FILES)
endif()
target_sources(${EXE_NAME} PRIVATE ${PPIC_QM_FILES})
target_link_libraries (${EXE_NAME} Qt${QT_VERSION_MAJOR}::Widgets Qt${QT_VERSION_MAJOR}::Svg Qt${QT_VERSION_MAJOR}::SvgWidgets)
if (exiv2_FOUND)
if(NOT TARGET Exiv2::exiv2lib AND TARGET exiv2lib)
# for exiv2 0.27.x and (macOS?) conan build
add_library(Exiv2::exiv2lib ALIAS exiv2lib)
endif()
target_link_libraries (${EXE_NAME} target_link_libraries (${EXE_NAME}
Exiv2::exiv2lib LibExiv2::LibExiv2
) )
target_compile_definitions(${EXE_NAME} PRIVATE target_compile_definitions(${EXE_NAME} PRIVATE
HAVE_EXIV2_VERSION="${exiv2_VERSION}" HAVE_EXIV2_VERSION="${LibExiv2_VERSION}"
) )
endif () endif ()
if (TARGET Qt6::DBus) if (Qt5DBus_FOUND OR Qt6DBus_FOUND)
target_link_libraries (${EXE_NAME} target_link_libraries (${EXE_NAME}
Qt${QT_VERSION_MAJOR}::DBus Qt${QT_VERSION_MAJOR}::DBus
) )
@ -144,6 +146,11 @@ endif()
# Extra build settings # Extra build settings
if (WIN32) if (WIN32)
set_property (
TARGET ${EXE_NAME}
PROPERTY WIN32_EXECUTABLE true
)
target_compile_definitions(${EXE_NAME} PRIVATE target_compile_definitions(${EXE_NAME} PRIVATE
FLAG_PORTABLE_MODE_SUPPORT=1 FLAG_PORTABLE_MODE_SUPPORT=1
) )
@ -174,7 +181,6 @@ macro (ppic_set_version_via_describe _describe_long)
endmacro () endmacro ()
# Version setup # Version setup
target_compile_definitions(${EXE_NAME} PRIVATE PPIC_VERSION_STRING="${CMAKE_PROJECT_VERSION}")
if (EXISTS "${CMAKE_SOURCE_DIR}/.git") if (EXISTS "${CMAKE_SOURCE_DIR}/.git")
find_package(Git) find_package(Git)
set_package_properties(Git PROPERTIES TYPE OPTIONAL PURPOSE "Determine exact build version.") set_package_properties(Git PROPERTIES TYPE OPTIONAL PURPOSE "Determine exact build version.")
@ -194,24 +200,8 @@ endif ()
# Install settings # Install settings
if (WIN32) if (WIN32)
set_target_properties(${EXE_NAME} PROPERTIES # TODO: try to avoid install to a "bin" subfolder under windows...
WIN32_EXECUTABLE TRUE # when fixed, don't forget to update the CI config file...
)
elseif (APPLE)
set_source_files_properties(assets/icons/app-icon.icns PROPERTIES
MACOSX_PACKAGE_LOCATION "Resources"
)
target_sources(${EXE_NAME} PUBLIC assets/icons/app-icon.icns)
# See https://cmake.org/cmake/help/v3.15/prop_tgt/MACOSX_BUNDLE_INFO_PLIST.html
set_target_properties(${EXE_NAME} PROPERTIES
MACOSX_BUNDLE TRUE
MACOSX_BUNDLE_INFO_PLIST ${CMAKE_CURRENT_SOURCE_DIR}/dist/MacOSXBundleInfo.plist.in
MACOSX_BUNDLE_BUNDLE_NAME "Pineapple Pictures"
MACOSX_BUNDLE_GUI_IDENTIFIER net.blumia.pineapple-pictures
MACOSX_BUNDLE_ICON_FILE app-icon.icns # contains the .icns file name, *without* the path.
MACOSX_BUNDLE_BUNDLE_VERSION ${PROJECT_VERSION}
MACOSX_BUNDLE_SHORT_VERSION_STRING ${PROJECT_VERSION_MAJOR}.${PROJECT_VERSION_MINOR}
)
elseif (UNIX) elseif (UNIX)
if (CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT) if (CMAKE_INSTALL_PREFIX_INITIALIZED_TO_DEFAULT)
set(CMAKE_INSTALL_PREFIX /usr) set(CMAKE_INSTALL_PREFIX /usr)
@ -238,7 +228,6 @@ elseif (UNIX)
endif() endif()
set (INSTALL_TARGETS_DEFAULT_ARGS set (INSTALL_TARGETS_DEFAULT_ARGS
BUNDLE DESTINATION .
RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR} RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}
LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR} LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}
ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR} COMPONENT Devel ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR} COMPONENT Devel
@ -249,25 +238,19 @@ install (
${INSTALL_TARGETS_DEFAULT_ARGS} ${INSTALL_TARGETS_DEFAULT_ARGS}
) )
if (TRANSLATION_RESOURCE_EMBEDDING) if (WIN32)
target_compile_definitions(${EXE_NAME}
PRIVATE TRANSLATION_RESOURCE_EMBEDDING
)
elseif (WIN32)
set (QM_FILE_INSTALL_DIR "${CMAKE_INSTALL_BINDIR}/translations") set (QM_FILE_INSTALL_DIR "${CMAKE_INSTALL_BINDIR}/translations")
else() else ()
set (QM_FILE_INSTALL_DIR "${CMAKE_INSTALL_FULL_DATADIR}/pineapple-pictures/translations") set (QM_FILE_INSTALL_DIR "${CMAKE_INSTALL_FULL_DATADIR}/pineapple-pictures/translations")
target_compile_definitions(${EXE_NAME} target_compile_definitions(${EXE_NAME}
PRIVATE QM_FILE_INSTALL_ABSOLUTE_DIR=${QM_FILE_INSTALL_DIR} PRIVATE QM_FILE_INSTALL_DIR=${QM_FILE_INSTALL_DIR}
) )
endif() endif ()
if (DEFINED QM_FILE_INSTALL_DIR) install (
install( FILES ${PPIC_QM_FILES}
FILES ${PPIC_QM_FILES} DESTINATION ${QM_FILE_INSTALL_DIR}
DESTINATION ${QM_FILE_INSTALL_DIR} )
)
endif()
feature_summary(WHAT ALL INCLUDE_QUIET_PACKAGES FATAL_ON_MISSING_REQUIRED_PACKAGES) feature_summary(WHAT ALL INCLUDE_QUIET_PACKAGES FATAL_ON_MISSING_REQUIRED_PACKAGES)
@ -283,7 +266,7 @@ elseif (APPLE)
# ... # ...
elseif (UNIX) elseif (UNIX)
set (CPACK_SYSTEM_NAME "${CMAKE_SYSTEM_NAME}-${CMAKE_SYSTEM_PROCESSOR}") set (CPACK_SYSTEM_NAME "${CMAKE_SYSTEM_NAME}-${CMAKE_SYSTEM_PROCESSOR}")
set (CPACK_DEBIAN_PACKAGE_SHLIBDEPS ON) set (CPACK_DEBIAN_PACKAGE_SHILIBDEPS ON)
set (CPACK_DEBIAN_PACKAGE_RECOMMENDS "kimageformat-plugins") set (CPACK_DEBIAN_PACKAGE_RECOMMENDS "kimageformat-plugins")
endif() endif()

42
LICENSE
View File

@ -1,21 +1,21 @@
MIT License MIT License
Copyright (c) 2025 BLumia Copyright (c) 2020 BLumia
Permission is hereby granted, free of charge, to any person obtaining a copy Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions: furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software. copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 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 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE. SOFTWARE.

11
LICENSES/BSD-3-Clause.txt Normal file
View File

@ -0,0 +1,11 @@
Copyright (c) <year> <owner>.
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.

169
NEWS
View File

@ -1,169 +0,0 @@
Version 1.1.0
~~~~~~~~~~~~~
Released: 2025-07-06
Features:
* New option to disable built-in close window animation
* New option to disable gallery looping
* Support load m3u8 as image gallery playlist
Miscellaneous:
* Drop Qt 5 support
Contributors:
Heimen Stoffels, albanobattistella, தமிழ்நேரம்
Version 1.0.0
~~~~~~~~~~~~~
Released: 2025-05-03
Features:
* Support enforces windowed mode on start-up
* Reload image automatically when current image gets updated
Bugfixes:
* Display correct text language on macOS
Miscellaneous:
* Use native text for shortcut editor's label
* Display native commandline message when possible
* Merge Qt translations into app applications as well
Contributors:
Heimen Stoffels, albanobattistella, mmahhi
Version 0.9.2
~~~~~~~~~~~~~
Released: 2025-03-05
Bugfixes:
* Refer to the right exiv2 CMake module so it can be found on Linux
Miscellaneous:
* Convert DEP5 to REUSE.toml for better REUSE compliance
* Update translations
Contributors:
Pino Toscano, TamilNeram
Version 0.9.1
~~~~~~~~~~~~~
Released: 2025-01-25
Features:
* Option to double-click to fullscreen
* Build-time option to embed translation resources
Bugfixes:
* Fix window size not adjusted when open file on macOS
* Should center window according to available screen geometry
Miscellaneous:
* Change close window bahavior on macOS
* Update translations
Contributors:
albanobattistella, Sabri Ünal
Version 0.9.0
~~~~~~~~~~~~~
Released: 2024-12-08
Features:
* Support custom shortcuts for existing actions
* Actions for frame-by-frame animated image playback support
Miscellaneous:
* Initial macOS bundle support
* bump minimum required CMake version to 3.16
* Update translations
Contributors:
albanobattistella, VenusGirl, gallegonovato, Sabri Ünal
Version 0.8.2.1
~~~~~~~~~~~~~
Released: 2024-10-27
Bugfixes:
* Cannot load translations caused by a change in 0.8.2
Version 0.8.2
~~~~~~~~~~~~~
Released: 2024-10-26
Features:
* New option to allow use light-color checkerboard by default
Contributors:
albanobattistella, mmahhi, gallegonovato
Version 0.8.1
~~~~~~~~~~~~~
Released: 2024-08-25
Features:
* New command line option to list all supported formats
Contributors:
albanobattistella, mmahhi, ovl-1, gallegonovato, Oğuz Ersen
Version 0.8.0
~~~~~~~~~~~~~
Released: 2024-06-29
Features:
* Support move image file to trash
Contributors:
albanobattistella, mmahhi, gallegonovato, Oğuz Ersen
Version 0.7.4
~~~~~~~~~~~~~
Released: 2024-04-04
Features:
* Add some icons for corresponding menu actions
Contributors:
Reza Almanda, mmahhi, Oğuz Ersen, volkov, Сергій
Version 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 automatically 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

141
README.md
View File

@ -1,63 +1,78 @@
Yet another image viewer. Yet another image viewer.
|CI|Build Status| |CI|Build Status|
|---|---| |---|---|
|Windows Build|[![Windows CI](https://github.com/BLumia/pineapple-pictures/actions/workflows/windows.yml/badge.svg)](https://github.com/BLumia/pineapple-pictures/actions/workflows/windows.yml)| |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)| |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)| |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) ![Pineapple Pictures - Main Window](https://repository-images.githubusercontent.com/211888654/e8697600-e370-11eb-9b2a-b71e05262954)
## Summary ## Summary
Pineapple Pictures is a lightweight image viewer that allows you view JPEG, PNG, GIF, SVG, PSD, KRA, XCF, TGA, HDR, 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. 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! ## Get it!
### Maintained by the original author - [GitHub Release Page](https://github.com/BLumia/pineapple-pictures/releases)
- [SourceForge](https://sourceforge.net/projects/pineapple-pictures/)
- [GitHub Release Page](https://github.com/BLumia/pineapple-pictures/releases) - Archlinux AUR: [pineapple-pictures](https://aur.archlinux.org/packages/pineapple-pictures/) | [pineapple-pictures-git](https://aur.archlinux.org/packages/pineapple-pictures-git/)
- [SourceForge](https://sourceforge.net/projects/pineapple-pictures/) - Debian (since bullseye) or Ubuntu (since 21.04): `sudo apt install pineapple-pictures`
- Archlinux AUR: [pineapple-pictures](https://aur.archlinux.org/packages/pineapple-pictures/) | [pineapple-pictures-git](https://aur.archlinux.org/packages/pineapple-pictures-git/) - [Itch.io Store](https://blumia.itch.io/pineapple-pictures)
- [Itch.io Store](https://blumia.itch.io/pineapple-pictures)
## Help Translation!
### Maintained by contributors / certain distro's package maintainers
[Translate this project on Weblate!](https://hosted.weblate.org/projects/pineapple-pictures/)
[![Packaging status](https://repology.org/badge/vertical-allrepos/pineapple-pictures.svg?columns=4)](https://repology.org/project/pineapple-pictures/versions)
## Build it manually:
## Help Translation!
Current state, we need:
[Translate this project on Weblate!](https://hosted.weblate.org/projects/pineapple-pictures/)
- `cmake`: as the build system.
## Build it manually: - `qt5` with `qt5-svg` and `qt5-tools`: since the app is using Qt.
- `libexiv2`: able to display more image metadata. (optional, but recommended)
Current state, we need:
Then we can build it with any proper c++ compiler like g++ or msvc.
- `cmake`: as the build system.
- `qt6` with `qt6-svg` and `qt6-tools`: since the app is using Qt. Building it just requires normal cmake building steps:
- `libexiv2`: able to display more image metadata. (optional, but recommended)
``` bash
Then we can build it with any proper c++ compiler like g++ or msvc. $ mkdir build && cd build
$ cmake ..
Building it just requires normal cmake building steps: $ cmake --build . # or simply using `make` if you are using Makefile as the cmake generator.
```
``` bash
$ mkdir build && cd build 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.
$ cmake ..
$ cmake --build . # or simply using `make` if you are using Makefile as the cmake generator. 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.
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.
> **Note**
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. > 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.
Image formats supports rely on Qt's imageformats plugins, just get the plugins you need from your distro's package manager will be fine. For Windows user, you may need build and install the imageformats plugin manually, read the content below. ### Linux
It's possible to build it under Windows, Linux, macOS, and maybe other desktop platforms that Qt is ported to. For platform specific build instructions, please read the [related wiki page](https://github.com/BLumia/pineapple-pictures/wiki/Platform-Specific-Build-Instructions). Just normal build process as other program will be fine. Nothing special ;)
> [!NOTE] For Archlinux there are also a [PKGBUILD](https://aur.archlinux.org/cgit/aur.git/tree/PKGBUILD?h=pineapple-pictures-git) you can use.
> 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.
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.
## License
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.
Pineapple Pictures as a whole is licensed under MIT license. Individual files may have a different, but compatible license.
### 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.

View File

@ -1,64 +1,78 @@
简单轻量的跨平台看图工具。 简单轻量的跨平台看图工具。
|CI|构建状态| |CI|构建状态|
|---|---| |---|---|
|Windows Build|[![Windows CI](https://github.com/BLumia/pineapple-pictures/actions/workflows/windows.yml/badge.svg)](https://github.com/BLumia/pineapple-pictures/actions/workflows/windows.yml)| |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)| |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)| |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) ![Pineapple Pictures - Main Window](https://repository-images.githubusercontent.com/211888654/e8697600-e370-11eb-9b2a-b71e05262954)
## 简介 ## 简介
菠萝看图是一个轻量图像查看器,允许你简单快捷的查看 JPEG, PNG, GIF, SVG, PSD, KRA, XCF, TGA, HDR, AVIF 等常用格式的图像文件,并提供了置顶窗口的选项以便你在使用其它软件时也可以将参考图片固定在顶端。 菠萝看图是一个轻量图像查看器,允许你简单快捷的查看 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/)
- [GitHub Release 页面](https://github.com/BLumia/pineapple-pictures/releases) | [gitee 发布页面](https://gitee.com/blumia/pineapple-pictures/releases) - Archlinux AUR: [pineapple-pictures](https://aur.archlinux.org/packages/pineapple-pictures/) | [pineapple-pictures-git](https://aur.archlinux.org/packages/pineapple-pictures-git/)
- [SourceForge](https://sourceforge.net/projects/pineapple-pictures/) - Debian (自 bullseye 起) 或 Ubuntu (自 21.04 起): `sudo apt install pineapple-pictures`
- Archlinux AUR: [pineapple-pictures](https://aur.archlinux.org/packages/pineapple-pictures/) | [pineapple-pictures-git](https://aur.archlinux.org/packages/pineapple-pictures-git/) - [Itch.io 商店](https://blumia.itch.io/pineapple-pictures)
- [Itch.io 商店](https://blumia.itch.io/pineapple-pictures)
- Flatpak (于 FlatHub): [net.blumia.pineapple-pictures](https://flathub.org/apps/net.blumia.pineapple-pictures) *([我应当使用 flatpak 版吗?](https://github.com/BLumia/pineapple-pictures/wiki/Container%E2%80%90based-Packaging-Solutions-Support))* ## 帮助翻译!
### 由贡献者/对应发行版的打包人员维护 [在 Weblate 上帮助此项目翻译到更多语言!](https://hosted.weblate.org/projects/pineapple-pictures/)
[![打包状态](https://repology.org/badge/vertical-allrepos/pineapple-pictures.svg?columns=4)](https://repology.org/project/pineapple-pictures/versions) ## 手动构建步骤:
## 帮助翻译! 当前状态,我们需要先确保如下依赖可用:
[在 Weblate 上帮助此项目翻译到更多语言!](https://hosted.weblate.org/projects/pineapple-pictures/) - `cmake`: 我们所使用的构建系统
- 包含 `qt5-svg``qt5-tools` 组件的 `qt5`: 此应用基于 Qt
## 手动构建步骤: - `libexiv2`: 用以获取和显示更多的图像元信息(可选,推荐)
当前状态,我们需要先确保如下依赖可用: 然后我们就可以使用任何常规的 c++ 编译器如 g++ 或 msvc 来进行构建了
- `cmake`: 我们所使用的构建系统 构建过程就是常规的 CMake 应用构建过程:
- 包含 `qt6-svg``qt6-tools` 组件的 `qt6`: 此应用基于 Qt
- `libexiv2`: 用以获取和显示更多的图像元信息(可选,推荐) ``` bash
$ mkdir build && cd build
然后我们就可以使用任何常规的 c++ 编译器如 g++ 或 msvc 来进行构建了 $ cmake ..
$ cmake --build . # 如果你使用 Makefile 作为 CMake 生成器,也可以直接简单的使用 `make`
构建过程就是常规的 CMake 应用构建过程: ```
``` bash 完毕后,一个名为 `ppic` 的可执行程序即会被生成以供使用。您也可以选择通过使用 CMake 生成的 `install` 目标继续将其安装到您的设备上(假设您使用 Makefile即可执行 `make install` 来进行安装)。构建步骤完毕后,您也可以使用 `cpack` 来对应用程序进行打包。
$ mkdir build && cd build
$ cmake .. 当 `exiv2` 在构建时可用时,此项目将尝试使用其进行构建,若您不希望使用 `exiv2`,请传递 `-DEXIV2_METADATA_SUPPORT=OFF` 参数给 `cmake`。此项目在找不到 `exiv2` 时并不会使用 `exiv2``EXIV2_METADATA_SUPPORT` 选项可供尽管存在可用的 `exiv2` 但您明确不希望启用其支持时使用。
$ cmake --build . # 如果你使用 Makefile 作为 CMake 生成器,也可以直接简单的使用 `make`
``` 此应用的图片格式支持依赖于 Qt 的 imageformats 插件,直接从您所用的发行版获取对应的图像格式插件即可。对于 Windows 用户,您可能需要手动构建和使用图像格式插件。下方给出了进一步的说明。
完毕后,一个名为 `ppic` 的可执行程序即会被生成以供使用。您也可以选择通过使用 CMake 生成的 `install` 目标继续将其安装到您的设备上(假设您使用 Makefile即可执行 `make install` 来进行安装)。构建步骤完毕后,您也可以使用 `cpack` 来对应用程序进行打包。 > **Note**
> 尽管存在一个可用于 QMake 构建的 `pineapple-pictures.pro` 文件,但其仅供简单测试所用且其并不包含 `exiv2` 支持。使用 QMake 构建此项目是 **不受支持** 的,请尽可能考虑使用 CMake。
当 `exiv2` 在构建时可用时,此项目将尝试使用其进行构建,若您不希望使用 `exiv2`,请传递 `-DEXIV2_METADATA_SUPPORT=OFF` 参数给 `cmake`。此项目在找不到 `exiv2` 时并不会使用 `exiv2``EXIV2_METADATA_SUPPORT` 选项可供尽管存在可用的 `exiv2` 但您明确不希望启用其支持时使用。
### Linux
此应用的图片格式支持依赖于 Qt 的 imageformats 插件,直接从您所用的发行版获取对应的图像格式插件即可。对于 Windows 用户,您可能需要手动构建和使用图像格式插件。下方给出了进一步的说明。
常规的构建步骤即可完成构建,不需要额外的处理步骤 ;)
在 Windows、Linux 以及 macOS 系统均可构建此应用,其它有移植 Qt 支持的平台也可能可以进行构建。若要了解一些平台相关的构建指引,请参阅[相关的 Wiki 页面](https://github.com/BLumia/pineapple-pictures/wiki/Platform-Specific-Build-Instructions)。
对于 Archlinux 发行版的用户,这里还有一个 [PKGBUILD](https://aur.archlinux.org/cgit/aur.git/tree/PKGBUILD?h=pineapple-pictures-git) 可供使用和参考。
> [!NOTE]
> 尽管存在一个可用于 QMake 构建的 `pineapple-pictures.pro` 文件,但其仅供简单测试所用且其并不包含 `exiv2` 支持。使用 QMake 构建此项目是 **不受支持** 的,请尽可能考虑使用 CMake 对于在基于 debian 的发行版中进行打包的需求, `CMakeLists.txt` 已经提供了一些基本的 cpack 配置以便生成一个有效的 `.deb` 软件包。在构建步骤完毕后,使用 `cpack -G DEB` 即可生成 DEB 软件包。您也可以参考 `.github/workflows/ubuntu.yml` 来查看当前正在使用 CI 配置是如何进行打包的
## 许可协议 目前,`DEB` 是当前唯一受到直接支持的 cpack 生成目标。若希望为此项目添加其它的 cpack 目标支持,欢迎发起合并请求。
菠萝看图整体使用 MIT 协议进行发布。项目所随的部分源文件可能具备不同但与之兼容的许可协议。 ### Windows
上述的构建步骤在 Windows 中也适用,但由于 Windows 中不具备类如大多 Linux 发行版中所提供的方便的软件包管理机制,故如果您需要任何 Qt 官方支持之外的图像格式例如 psdxcfkra 等格式的支持,你就可能需要自行获取并构建对应的 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 协议进行发布。项目所随的部分源文件可能具备不同但与之兼容的许可协议。

View File

@ -1,33 +0,0 @@
version = 1
SPDX-PackageName = "Pineapple Pictures"
SPDX-PackageDownloadLocation = "https://github.com/BLumia/pineapple-pictures"
[[annotations]]
path = [".gitattributes", ".git-blame-ignore-revs", ".gitignore", "appveyor.yml", ".github/**"]
precedence = "aggregate"
SPDX-FileCopyrightText = "None"
SPDX-License-Identifier = "CC0-1.0"
[[annotations]]
path = ["README**.md", "NEWS", "assets/**.rc", "assets/**.qrc", "dist/**"]
precedence = "aggregate"
SPDX-FileCopyrightText = "None"
SPDX-License-Identifier = "CC0-1.0"
[[annotations]]
path = ["app/translations/**.ts", "assets/plain/translators.html"]
precedence = "aggregate"
SPDX-FileCopyrightText = "Translators from hosted.weblate.org"
SPDX-License-Identifier = "MIT"
[[annotations]]
path = "assets/icons/**.svg"
precedence = "aggregate"
SPDX-FileCopyrightText = "2022 Gary Wang"
SPDX-License-Identifier = "MIT"
[[annotations]]
path = "assets/icons/app-icon.**"
precedence = "aggregate"
SPDX-FileCopyrightText = "2020 Lovelyblack"
SPDX-License-Identifier = "MIT"

View File

@ -1,181 +1,172 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "aboutdialog.h" #include "aboutdialog.h"
#include <QAbstractButton> #include <QAbstractButton>
#include <QDialogButtonBox> #include <QDialogButtonBox>
#include <QTextBrowser> #include <QTextBrowser>
#include <QTabWidget> #include <QTabWidget>
#include <QVBoxLayout> #include <QVBoxLayout>
#include <QApplication> #include <QApplication>
#include <QLabel> #include <QLabel>
#include <QPushButton> #include <QPushButton>
#include <QFile> #include <QFile>
using namespace Qt::Literals::StringLiterals; AboutDialog::AboutDialog(QWidget *parent)
: QDialog(parent)
AboutDialog::AboutDialog(QWidget *parent) , m_tabWidget(new QTabWidget)
: QDialog(parent) , m_buttonBox(new QDialogButtonBox)
, m_tabWidget(new QTabWidget) , m_helpTextEdit(new QTextBrowser)
, m_buttonBox(new QDialogButtonBox) , m_aboutTextEdit(new QTextBrowser)
, m_helpTextEdit(new QTextBrowser) , m_specialThanksTextEdit(new QTextBrowser)
, m_aboutTextEdit(new QTextBrowser) , m_licenseTextEdit(new QTextBrowser)
, m_specialThanksTextEdit(new QTextBrowser) , m_3rdPartyLibsTextEdit(new QTextBrowser)
, m_licenseTextEdit(new QTextBrowser) {
, m_3rdPartyLibsTextEdit(new QTextBrowser) this->setWindowTitle(tr("About"));
{
this->setWindowTitle(tr("About")); const QStringList helpStr {
QStringLiteral("<p>%1</p>").arg(tr("Launch application with image file path as argument to load the file.")),
const QStringList helpStr { QStringLiteral("<p>%1</p>").arg(tr("Drag and drop image file onto the window is also supported.")),
u"<p>%1</p>"_s.arg(tr("Launch application with image file path as argument to load the file.")), QStringLiteral("<p>%1</p>").arg(tr("None of the operations in this application will alter the pictures on disk.")),
u"<p>%1</p>"_s.arg(tr("Drag and drop image file onto the window is also supported.")), QStringLiteral("<p>%1</p>").arg(tr("Context menu option explanation:")),
u"<p>%1</p>"_s.arg(tr("None of the operations in this application will alter the pictures on disk.")), QStringLiteral("<ul>"),
u"<p>%1</p>"_s.arg(tr("Context menu option explanation:")), // blumia: Chain two arg() here since it seems lupdate will remove one of them if we use
u"<ul>"_s, // the old `arg(QCoreApp::translate(), tr())` way, but it's worth to mention
// blumia: Chain two arg() here since it seems lupdate will remove one of them if we use // `arg(QCoreApp::translate(), this->tr())` works, but lupdate will complain about the usage.
// the old `arg(QCoreApp::translate(), tr())` way, but it's worth to mention QStringLiteral("<li><b>%1</b>:<br/>%2</li>")
// `arg(QCoreApp::translate(), this->tr())` works, but lupdate will complain about the usage. .arg(QCoreApplication::translate("MainWindow", "Stay on top"))
u"<li><b>%1</b>:<br/>%2</li>"_s .arg(tr("Make window stay on top of all other windows.")),
.arg(QCoreApplication::translate("MainWindow", "Stay on top")) QStringLiteral("<li><b>%1</b>:<br/>%2</li>")
.arg(tr("Make window stay on top of all other windows.")), .arg(QCoreApplication::translate("MainWindow", "Protected mode"))
u"<li><b>%1</b>:<br/>%2</li>"_s .arg(tr("Avoid close window accidentally. (eg. by double clicking the window)")),
.arg(QCoreApplication::translate("MainWindow", "Protected mode")) QStringLiteral("</ul>")
.arg(tr("Avoid close window accidentally. (eg. by double clicking the window)")), };
u"<li><b>%1</b>:<br/>%2</li>"_s
.arg(QCoreApplication::translate("MainWindow", "Keep transformation", "The 'transformation' means the flip/rotation status that currently applied to the image view")) const QStringList aboutStr {
.arg(tr("Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images.")), QStringLiteral("<center><img width='128' height='128' src=':/icons/app-icon.svg'/><br/>"),
u"</ul>"_s qApp->applicationDisplayName(),
}; #ifdef GIT_DESCRIBE_VERSION_STRING
(QStringLiteral("<br/>") + tr("Version: %1").arg(GIT_DESCRIBE_VERSION_STRING)),
const QStringList aboutStr { #endif // GIT_DESCRIBE_VERSION_STRING
u"<center><img width='128' height='128' src=':/icons/app-icon.svg'/><br/>"_s, QStringLiteral("<hr/>"),
qApp->applicationDisplayName(), tr("Copyright (c) %1 %2", "%1 is year, %2 is the name of copyright holder(s)")
(u"<br/>"_s + tr("Version: %1").arg( .arg(QStringLiteral("2023"), QStringLiteral("<a href='https://github.com/BLumia'>@BLumia</a>")),
#ifdef GIT_DESCRIBE_VERSION_STRING QStringLiteral("<br/>"),
GIT_DESCRIBE_VERSION_STRING tr("Logo designed by %1").arg(QStringLiteral("<a href='https://github.com/Lovelyblack'>@Lovelyblack</a>")),
#else QStringLiteral("<hr/>"),
qApp->applicationVersion() tr("Built with Qt %1 (%2)").arg(QT_VERSION_STR, QSysInfo::buildCpuArchitecture()),
#endif // GIT_DESCRIBE_VERSION_STRING QStringLiteral("<br/><a href='%1'>%2</a>").arg("https://github.com/BLumia/pineapple-pictures", tr("Source code")),
)), QStringLiteral("</center>")
u"<hr/>"_s, };
tr("Copyright (c) %1 %2", "%1 is year, %2 is the name of copyright holder(s)")
.arg(u"2025"_s, u"<a href='https://github.com/BLumia'>@BLumia</a>"_s), QFile translaterHtml(":/plain/translators.html");
u"<br/>"_s, bool canOpenFile = translaterHtml.open(QIODevice::ReadOnly);
tr("Logo designed by %1").arg(u"<a href='https://github.com/Lovelyblack'>@Lovelyblack</a>"_s), const QByteArray & translatorList = canOpenFile ? translaterHtml.readAll() : "";
u"<hr/>"_s,
tr("Built with Qt %1 (%2)").arg(QT_VERSION_STR, QSysInfo::buildCpuArchitecture()), const QStringList specialThanksStr {
QStringLiteral("<br/><a href='%1'>%2</a>").arg("https://github.com/BLumia/pineapple-pictures", tr("Source code")), QStringLiteral("<h1 align='center'>%1</h1><a href='%2'>%3</a><p>%4</p>").arg(
u"</center>"_s tr("Contributors"),
}; QStringLiteral("https://github.com/BLumia/pineapple-pictures/graphs/contributors"),
tr("List of contributors on GitHub"),
QFile translaterHtml(u":/plain/translators.html"_s); tr("Thanks to all people who contributed to this project.")
bool canOpenFile = translaterHtml.open(QIODevice::ReadOnly); ),
const QByteArray & translatorList = canOpenFile ? translaterHtml.readAll() : QByteArrayLiteral("");
QStringLiteral("<h1 align='center'>%1</h1><p>%2</p>%3").arg(
const QStringList specialThanksStr { tr("Translators"),
u"<h1 align='center'>%1</h1><a href='%2'>%3</a><p>%4</p>"_s.arg( tr("I would like to thank the following people who volunteered to translate this application."),
tr("Contributors"), translatorList
u"https://github.com/BLumia/pineapple-pictures/graphs/contributors"_s, )
tr("List of contributors on GitHub"), };
tr("Thanks to all people who contributed to this project.")
), const QStringList licenseStr {
QStringLiteral("<h1 align='center'><b>%1</b></h1>").arg(tr("Your Rights")),
u"<h1 align='center'>%1</h1><p>%2</p>%3"_s.arg( QStringLiteral("<p>%1</p><p>%2</p><ul><li>%3</li><li>%4</li><li>%5</li><li>%6</li></ul>").arg(
tr("Translators"), tr("%1 is released under the MIT License."), // %1
tr("I would like to thank the following people who volunteered to translate this application."), tr("This license grants people a number of freedoms:"), // %2
translatorList 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
const QStringList licenseStr { ).arg(QStringLiteral("<i>%1</i>")),
u"<h1 align='center'><b>%1</b></h1>"_s.arg(tr("Your Rights")), QStringLiteral("<p>%1</p>").arg(tr("The MIT license guarantees you this freedom. Nobody is ever permitted to take it away.")),
u"<p>%1</p><p>%2</p><ul><li>%3</li><li>%4</li><li>%5</li><li>%6</li></ul>"_s.arg( QStringLiteral("<hr/><pre>%2</pre>")
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 const QString mitLicense(QStringLiteral(R"(Expat/MIT License
tr("You are free to distribute %1"), // %4
tr("You can study how %1 works and change it"), // %5 Copyright (c) 2023 BLumia
tr("You can distribute changed versions of %1") // %6
).arg(u"<i>%1</i>"_s), Permission is hereby granted, free of charge, to any person obtaining a copy
u"<p>%1</p>"_s.arg(tr("The MIT license guarantees you this freedom. Nobody is ever permitted to take it away.")), of this software and associated documentation files (the "Software"), to deal
u"<hr/><pre>%2</pre>"_s 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
const QString mitLicense(QStringLiteral(R"(Expat/MIT License furnished to do so, subject to the following conditions:
Copyright (c) 2025 BLumia The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
in the Software without restriction, including without limitation the rights IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
copies of the Software, and to permit persons to whom the Software is AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
furnished to do so, subject to the following conditions: 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
The above copyright notice and this permission notice shall be included in all SOFTWARE.
copies or substantial portions of the Software. )"));
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR const QStringList thirdPartyLibsStr {
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, QStringLiteral("<h1 align='center'><b>%1</b></h1>").arg(tr("Third-party Libraries used by %1")),
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE tr("%1 is built on the following free software libraries:", "Free as in freedom"),
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER QStringLiteral("<ul>"),
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, #ifdef HAVE_EXIV2_VERSION
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE QStringLiteral("<li><a href='%1'>%2</a>: %3</li>").arg("https://www.exiv2.org/", "Exiv2", "GPLv2"),
SOFTWARE. #endif // EXIV2_VERSION
)")); QStringLiteral("<li><a href='%1'>%2</a>: %3</li>").arg("https://www.qt.io/", "Qt", "GPLv2 + GPLv3 + LGPLv2.1 + LGPLv3"),
QStringLiteral("</ul>")
const QStringList thirdPartyLibsStr { };
u"<h1 align='center'><b>%1</b></h1>"_s.arg(tr("Third-party Libraries used by %1")),
tr("%1 is built on the following free software libraries:", "Free as in freedom"), m_helpTextEdit->setText(helpStr.join('\n'));
u"<ul>"_s,
#ifdef HAVE_EXIV2_VERSION m_aboutTextEdit->setText(aboutStr.join('\n'));
u"<li><a href='%1'>%2</a>: %3</li>"_s.arg("https://www.exiv2.org/", "Exiv2", "GPLv2"), m_aboutTextEdit->setOpenExternalLinks(true);
#endif // EXIV2_VERSION
u"<li><a href='%1'>%2</a>: %3</li>"_s.arg("https://www.qt.io/", "Qt", "GPLv2 + GPLv3 + LGPLv2.1 + LGPLv3"), m_specialThanksTextEdit->setText(specialThanksStr.join('\n'));
u"</ul>"_s m_specialThanksTextEdit->setOpenExternalLinks(true);
};
m_licenseTextEdit->setText(licenseStr.join('\n').arg(qApp->applicationDisplayName(), mitLicense));
m_helpTextEdit->setText(helpStr.join('\n'));
m_3rdPartyLibsTextEdit->setText(thirdPartyLibsStr.join('\n').arg(QStringLiteral("<i>%1</i>").arg(qApp->applicationDisplayName())));
m_aboutTextEdit->setText(aboutStr.join('\n')); m_3rdPartyLibsTextEdit->setOpenExternalLinks(true);
m_aboutTextEdit->setOpenExternalLinks(true);
m_tabWidget->addTab(m_helpTextEdit, tr("&Help"));
m_specialThanksTextEdit->setText(specialThanksStr.join('\n')); m_tabWidget->addTab(m_aboutTextEdit, tr("&About"));
m_specialThanksTextEdit->setOpenExternalLinks(true); m_tabWidget->addTab(m_specialThanksTextEdit, tr("&Special Thanks"));
m_tabWidget->addTab(m_licenseTextEdit, tr("&License"));
m_licenseTextEdit->setText(licenseStr.join('\n').arg(qApp->applicationDisplayName(), mitLicense)); m_tabWidget->addTab(m_3rdPartyLibsTextEdit, tr("&Third-party Libraries"));
m_3rdPartyLibsTextEdit->setText(thirdPartyLibsStr.join('\n').arg(u"<i>%1</i>"_s).arg(qApp->applicationDisplayName())); m_buttonBox->setStandardButtons(QDialogButtonBox::Close);
m_3rdPartyLibsTextEdit->setOpenExternalLinks(true); connect(m_buttonBox, QOverload<QAbstractButton *>::of(&QDialogButtonBox::clicked), this, [this](){
this->close();
m_tabWidget->addTab(m_helpTextEdit, tr("&Help")); });
m_tabWidget->addTab(m_aboutTextEdit, tr("&About"));
m_tabWidget->addTab(m_specialThanksTextEdit, tr("&Special Thanks")); setLayout(new QVBoxLayout);
m_tabWidget->addTab(m_licenseTextEdit, tr("&License"));
m_tabWidget->addTab(m_3rdPartyLibsTextEdit, tr("&Third-party Libraries")); layout()->addWidget(m_tabWidget);
layout()->addWidget(m_buttonBox);
m_buttonBox->setStandardButtons(QDialogButtonBox::Close);
connect(m_buttonBox, QOverload<QAbstractButton *>::of(&QDialogButtonBox::clicked), this, [this](){ setMinimumSize(361, 161); // not sure why it complain "Unable to set geometry"
this->close(); setWindowFlag(Qt::WindowContextHelpButtonHint, false);
}); }
setLayout(new QVBoxLayout); AboutDialog::~AboutDialog()
{
layout()->addWidget(m_tabWidget);
layout()->addWidget(m_buttonBox); }
setMinimumSize(361, 161); // not sure why it complain "Unable to set geometry" QSize AboutDialog::sizeHint() const
setWindowFlag(Qt::WindowContextHelpButtonHint, false); {
} return QSize(520, 350);
}
AboutDialog::~AboutDialog()
{
}
QSize AboutDialog::sizeHint() const
{
return QSize(520, 350);
}

View File

@ -1,36 +1,36 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#ifndef ABOUTDIALOG_H #ifndef ABOUTDIALOG_H
#define ABOUTDIALOG_H #define ABOUTDIALOG_H
#include <QDialog> #include <QDialog>
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE
class QTextBrowser; class QTextBrowser;
class QTabWidget; class QTabWidget;
class QDialogButtonBox; class QDialogButtonBox;
QT_END_NAMESPACE QT_END_NAMESPACE
class AboutDialog : public QDialog class AboutDialog : public QDialog
{ {
Q_OBJECT Q_OBJECT
public: public:
explicit AboutDialog(QWidget *parent = nullptr); explicit AboutDialog(QWidget *parent = nullptr);
~AboutDialog() override; ~AboutDialog() override;
QSize sizeHint() const override; QSize sizeHint() const override;
private: private:
QTabWidget * m_tabWidget = nullptr; QTabWidget * m_tabWidget = nullptr;
QDialogButtonBox * m_buttonBox = nullptr; QDialogButtonBox * m_buttonBox = nullptr;
QTextBrowser * m_helpTextEdit = nullptr; QTextBrowser * m_helpTextEdit = nullptr;
QTextBrowser * m_aboutTextEdit = nullptr; QTextBrowser * m_aboutTextEdit = nullptr;
QTextBrowser * m_specialThanksTextEdit = nullptr; QTextBrowser * m_specialThanksTextEdit = nullptr;
QTextBrowser * m_licenseTextEdit = nullptr; QTextBrowser * m_licenseTextEdit = nullptr;
QTextBrowser * m_3rdPartyLibsTextEdit = nullptr; QTextBrowser * m_3rdPartyLibsTextEdit = nullptr;
}; };
#endif // ABOUTDIALOG_H #endif // ABOUTDIALOG_H

View File

@ -1,153 +1,150 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "actionmanager.h" #include "actionmanager.h"
#include "mainwindow.h" #include "mainwindow.h"
#include <QGuiApplication> #include <QGuiApplication>
#include <QSvgRenderer> #include <QSvgRenderer>
#include <QPainter> #include <QPainter>
#define ICON_NAME(name)\ #define ICON_NAME(name)\
QStringLiteral(":/icons/" #name ".svg") QStringLiteral(":/icons/" #name ".svg")
#define ACTION_NAME(s) QStringLiteral(STRIFY(s)) #define ACTION_NAME(s) QStringLiteral(STRIFY(s))
#define STRIFY(s) #s #define STRIFY(s) #s
QIcon ActionManager::loadHidpiIcon(const QString &resp, QSize sz) ActionManager::ActionManager()
{ {
QSvgRenderer r(resp);
QPixmap pm = QPixmap(sz * qApp->devicePixelRatio()); }
pm.fill(Qt::transparent);
QPainter p(&pm); ActionManager::~ActionManager()
r.render(&p); {
pm.setDevicePixelRatio(qApp->devicePixelRatio());
return QIcon(pm); }
}
QIcon ActionManager::loadHidpiIcon(const QString &resp, QSize sz)
void ActionManager::setupAction(MainWindow *mainWindow) {
{ QSvgRenderer r(resp);
auto create_action = [] (QWidget *w, QAction **a, QString i, QString an, bool iconFromTheme = false) { QPixmap pm = QPixmap(sz * qApp->devicePixelRatio());
*a = new QAction(w); pm.fill(Qt::transparent);
if (!i.isNull()) QPainter p(&pm);
(*a)->setIcon(iconFromTheme ? QIcon::fromTheme(i) : ActionManager::loadHidpiIcon(i)); r.render(&p);
(*a)->setObjectName(an); pm.setDevicePixelRatio(qApp->devicePixelRatio());
w->addAction(*a); return QIcon(pm);
}; }
#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); void ActionManager::setupAction(MainWindow *mainWindow)
CREATE_NEW_ICON_ACTION(mainWindow, actionToggleMaximize, view-fullscreen); {
CREATE_NEW_ICON_ACTION(mainWindow, actionZoomIn, zoom-in); auto create_action = [] (QWidget *w, QAction **a, QString i, QString an) {
CREATE_NEW_ICON_ACTION(mainWindow, actionZoomOut, zoom-out); *a = new QAction(w);
CREATE_NEW_ICON_ACTION(mainWindow, actionToggleCheckerboard, view-background-checkerboard); if (!i.isNull())
CREATE_NEW_ICON_ACTION(mainWindow, actionRotateClockwise, object-rotate-right); (*a)->setIcon(ActionManager::loadHidpiIcon(i));
#undef CREATE_NEW_ICON_ACTION (*a)->setObjectName(an);
w->addAction(*a);
#define CREATE_NEW_ACTION(w, a) create_action(w, &a, QString(), ACTION_NAME(a)) };
#define CREATE_NEW_THEMEICON_ACTION(w, a, i) create_action(w, &a, QLatin1String(STRIFY(i)), ACTION_NAME(a), true) #define CREATE_NEW_ICON_ACTION(w, a, i) create_action(w, &a, ICON_NAME(i), ACTION_NAME(a))
CREATE_NEW_ACTION(mainWindow, actionRotateCounterClockwise); CREATE_NEW_ICON_ACTION(mainWindow, actionActualSize, zoom-original);
CREATE_NEW_ACTION(mainWindow, actionPrevPicture); CREATE_NEW_ICON_ACTION(mainWindow, actionToggleMaximize, view-fullscreen);
CREATE_NEW_ACTION(mainWindow, actionNextPicture); CREATE_NEW_ICON_ACTION(mainWindow, actionZoomIn, zoom-in);
CREATE_NEW_ICON_ACTION(mainWindow, actionZoomOut, zoom-out);
CREATE_NEW_ACTION(mainWindow, actionTogglePauseAnimation); CREATE_NEW_ICON_ACTION(mainWindow, actionToggleCheckerboard, view-background-checkerboard);
CREATE_NEW_ACTION(mainWindow, actionAnimationNextFrame); CREATE_NEW_ICON_ACTION(mainWindow, actionRotateClockwise, object-rotate-right);
#undef CREATE_NEW_ICON_ACTION
CREATE_NEW_THEMEICON_ACTION(mainWindow, actionOpen, document-open);
CREATE_NEW_ACTION(mainWindow, actionHorizontalFlip); #define CREATE_NEW_ACTION(w, a) create_action(w, &a, QString(), ACTION_NAME(a))
CREATE_NEW_ACTION(mainWindow, actionFitInView); CREATE_NEW_ACTION(mainWindow, actionPrevPicture);
CREATE_NEW_ACTION(mainWindow, actionFitByWidth); CREATE_NEW_ACTION(mainWindow, actionNextPicture);
CREATE_NEW_THEMEICON_ACTION(mainWindow, actionCopyPixmap, edit-copy);
CREATE_NEW_ACTION(mainWindow, actionCopyFilePath); CREATE_NEW_ACTION(mainWindow, actionOpen);
CREATE_NEW_THEMEICON_ACTION(mainWindow, actionPaste, edit-paste); CREATE_NEW_ACTION(mainWindow, actionHorizontalFlip);
CREATE_NEW_THEMEICON_ACTION(mainWindow, actionTrash, edit-delete); CREATE_NEW_ACTION(mainWindow, actionFitInView);
CREATE_NEW_ACTION(mainWindow, actionToggleStayOnTop); CREATE_NEW_ACTION(mainWindow, actionFitByWidth);
CREATE_NEW_ACTION(mainWindow, actionToggleProtectMode); CREATE_NEW_ACTION(mainWindow, actionCopyPixmap);
CREATE_NEW_ACTION(mainWindow, actionToggleAvoidResetTransform); CREATE_NEW_ACTION(mainWindow, actionCopyFilePath);
CREATE_NEW_ACTION(mainWindow, actionSettings); CREATE_NEW_ACTION(mainWindow, actionPaste);
CREATE_NEW_THEMEICON_ACTION(mainWindow, actionHelp, system-help); CREATE_NEW_ACTION(mainWindow, actionToggleStayOnTop);
CREATE_NEW_THEMEICON_ACTION(mainWindow, actionLocateInFileManager, system-file-manager); CREATE_NEW_ACTION(mainWindow, actionToggleProtectMode);
CREATE_NEW_THEMEICON_ACTION(mainWindow, actionProperties, document-properties); CREATE_NEW_ACTION(mainWindow, actionToggleAvoidResetTransform);
CREATE_NEW_ACTION(mainWindow, actionQuitApp); CREATE_NEW_ACTION(mainWindow, actionSettings);
#undef CREATE_NEW_ACTION CREATE_NEW_ACTION(mainWindow, actionHelp);
#undef CREATE_NEW_THEMEICON_ACTION CREATE_NEW_ACTION(mainWindow, actionLocateInFileManager);
CREATE_NEW_ACTION(mainWindow, actionProperties);
retranslateUi(mainWindow); CREATE_NEW_ACTION(mainWindow, actionQuitApp);
#undef CREATE_NEW_ACTION
QMetaObject::connectSlotsByName(mainWindow);
} retranslateUi(mainWindow);
void ActionManager::retranslateUi(MainWindow *mainWindow) QMetaObject::connectSlotsByName(mainWindow);
{ }
Q_UNUSED(mainWindow);
void ActionManager::retranslateUi(MainWindow *mainWindow)
actionOpen->setText(QCoreApplication::translate("MainWindow", "&Open...", nullptr)); {
Q_UNUSED(mainWindow);
actionActualSize->setText(QCoreApplication::translate("MainWindow", "Actual size", nullptr));
actionToggleMaximize->setText(QCoreApplication::translate("MainWindow", "Toggle maximize", nullptr)); actionOpen->setText(QCoreApplication::translate("MainWindow", "&Open...", nullptr));
actionZoomIn->setText(QCoreApplication::translate("MainWindow", "Zoom in", nullptr));
actionZoomOut->setText(QCoreApplication::translate("MainWindow", "Zoom out", nullptr)); actionActualSize->setText(QCoreApplication::translate("MainWindow", "Actual size", nullptr));
actionToggleCheckerboard->setText(QCoreApplication::translate("MainWindow", "Toggle Checkerboard", nullptr)); actionToggleMaximize->setText(QCoreApplication::translate("MainWindow", "Toggle maximize", nullptr));
actionRotateClockwise->setText(QCoreApplication::translate("MainWindow", "Rotate right", nullptr)); actionZoomIn->setText(QCoreApplication::translate("MainWindow", "Zoom in", nullptr));
actionRotateCounterClockwise->setText(QCoreApplication::translate("MainWindow", "Rotate left", nullptr)); actionZoomOut->setText(QCoreApplication::translate("MainWindow", "Zoom out", nullptr));
actionToggleCheckerboard->setText(QCoreApplication::translate("MainWindow", "Toggle Checkerboard", nullptr));
actionPrevPicture->setText(QCoreApplication::translate("MainWindow", "Previous image", nullptr)); actionRotateClockwise->setText(QCoreApplication::translate("MainWindow", "Rotate right", nullptr));
actionNextPicture->setText(QCoreApplication::translate("MainWindow", "Next image", nullptr));
actionPrevPicture->setText(QCoreApplication::translate("MainWindow", "Previous image", nullptr));
actionTogglePauseAnimation->setText(QCoreApplication::translate("MainWindow", "Pause/Resume Animation", nullptr)); actionNextPicture->setText(QCoreApplication::translate("MainWindow", "Next image", nullptr));
actionAnimationNextFrame->setText(QCoreApplication::translate("MainWindow", "Animation Go to Next Frame", nullptr));
actionHorizontalFlip->setText(QCoreApplication::translate("MainWindow", "Flip &Horizontally", nullptr));
actionHorizontalFlip->setText(QCoreApplication::translate("MainWindow", "Flip &Horizontally", nullptr)); actionFitInView->setText("Fit in view"); // TODO: what should it called?
actionFitInView->setText(QCoreApplication::translate("MainWindow", "Fit to view", nullptr)); actionFitByWidth->setText("Fit by width"); // TODO: what should it called?
actionFitByWidth->setText(QCoreApplication::translate("MainWindow", "Fit to width", nullptr)); actionCopyPixmap->setText(QCoreApplication::translate("MainWindow", "Copy P&ixmap", nullptr));
actionCopyPixmap->setText(QCoreApplication::translate("MainWindow", "Copy P&ixmap", nullptr)); actionCopyFilePath->setText(QCoreApplication::translate("MainWindow", "Copy &File Path", nullptr));
actionCopyFilePath->setText(QCoreApplication::translate("MainWindow", "Copy &File Path", nullptr)); actionPaste->setText(QCoreApplication::translate("MainWindow", "&Paste", nullptr));
actionPaste->setText(QCoreApplication::translate("MainWindow", "&Paste", nullptr)); actionToggleStayOnTop->setText(QCoreApplication::translate("MainWindow", "Stay on top", nullptr));
actionTrash->setText(QCoreApplication::translate("MainWindow", "Move to Trash", nullptr)); actionToggleProtectMode->setText(QCoreApplication::translate("MainWindow", "Protected mode", nullptr));
actionToggleStayOnTop->setText(QCoreApplication::translate("MainWindow", "Stay on top", nullptr)); actionToggleAvoidResetTransform->setText("Avoid reset transform"); // TODO: what should it called?
actionToggleProtectMode->setText(QCoreApplication::translate("MainWindow", "Protected mode", nullptr)); actionSettings->setText(QCoreApplication::translate("MainWindow", "Configure...", nullptr));
actionToggleAvoidResetTransform->setText(QCoreApplication::translate("MainWindow", "Keep transformation", "The 'transformation' means the flip/rotation status that currently applied to the image view")); actionHelp->setText(QCoreApplication::translate("MainWindow", "Help", nullptr));
actionSettings->setText(QCoreApplication::translate("MainWindow", "Configure...", nullptr)); #ifdef Q_OS_WIN
actionHelp->setText(QCoreApplication::translate("MainWindow", "Help", nullptr)); actionLocateInFileManager->setText(
#ifdef Q_OS_WIN QCoreApplication::translate(
actionLocateInFileManager->setText( "MainWindow", "Show in File Explorer",
QCoreApplication::translate( "File Explorer is the name of explorer.exe under Windows"
"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));
#else #endif // Q_OS_WIN
actionLocateInFileManager->setText(QCoreApplication::translate("MainWindow", "Show in directory", nullptr)); actionProperties->setText(QCoreApplication::translate("MainWindow", "Properties", nullptr));
#endif // Q_OS_WIN actionQuitApp->setText(QCoreApplication::translate("MainWindow", "Quit", nullptr));
actionProperties->setText(QCoreApplication::translate("MainWindow", "Properties", nullptr)); }
actionQuitApp->setText(QCoreApplication::translate("MainWindow", "Quit", nullptr));
} void ActionManager::setupShortcuts()
{
void ActionManager::setupShortcuts() actionOpen->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_O));
{ actionActualSize->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_0));
actionOpen->setShortcut(QKeySequence::Open); actionZoomIn->setShortcut(QKeySequence(QKeySequence::ZoomIn));
actionActualSize->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_0)); actionZoomOut->setShortcut(QKeySequence(QKeySequence::ZoomOut));
actionZoomIn->setShortcut(QKeySequence::ZoomIn); actionPrevPicture->setShortcuts({
actionZoomOut->setShortcut(QKeySequence::ZoomOut); QKeySequence(Qt::Key_PageUp),
actionPrevPicture->setShortcuts({ QKeySequence(Qt::Key_Left),
QKeySequence(Qt::Key_PageUp), });
QKeySequence(Qt::Key_Left), actionNextPicture->setShortcuts({
}); QKeySequence(Qt::Key_PageDown),
actionNextPicture->setShortcuts({ QKeySequence(Qt::Key_Right),
QKeySequence(Qt::Key_PageDown), });
QKeySequence(Qt::Key_Right), actionHorizontalFlip->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_R));
}); actionCopyPixmap->setShortcut(QKeySequence(QKeySequence::Copy));
actionHorizontalFlip->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_R)); actionPaste->setShortcut(QKeySequence::Paste);
actionCopyPixmap->setShortcut(QKeySequence::Copy); actionHelp->setShortcut(QKeySequence::HelpContents);
actionPaste->setShortcut(QKeySequence::Paste); actionSettings->setShortcut(QKeySequence::Preferences);
actionTrash->setShortcut(QKeySequence::Delete); actionProperties->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_I));
actionHelp->setShortcut(QKeySequence::HelpContents); actionQuitApp->setShortcuts({
actionSettings->setShortcut(QKeySequence::Preferences); QKeySequence(Qt::Key_Space),
actionProperties->setShortcut(QKeySequence(Qt::CTRL | Qt::Key_I)); QKeySequence(Qt::Key_Escape)
actionQuitApp->setShortcuts({ });
QKeySequence(Qt::Key_Space), }
QKeySequence(Qt::Key_Escape)
});
}

View File

@ -1,58 +1,53 @@
// SPDX-FileCopyrightText: 2025 Gary Wang <git@blumia.net> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#ifndef ACTIONMANAGER_H #ifndef ACTIONMANAGER_H
#define ACTIONMANAGER_H #define ACTIONMANAGER_H
#include <QAction> #include <QAction>
class MainWindow; class MainWindow;
class ActionManager class ActionManager
{ {
public: public:
explicit ActionManager() = default; ActionManager();
~ActionManager() = default; ~ActionManager();
void setupAction(MainWindow * mainWindow); void setupAction(MainWindow * mainWindow);
void retranslateUi(MainWindow *MainWindow); void retranslateUi(MainWindow *MainWindow);
void setupShortcuts(); void setupShortcuts();
static QIcon loadHidpiIcon(const QString &resp, QSize sz = QSize(32, 32)); static QIcon loadHidpiIcon(const QString &resp, QSize sz = QSize(32, 32));
public: public:
QAction *actionOpen; QAction *actionOpen;
QAction *actionActualSize; QAction *actionActualSize;
QAction *actionToggleMaximize; QAction *actionToggleMaximize;
QAction *actionZoomIn; QAction *actionZoomIn;
QAction *actionZoomOut; QAction *actionZoomOut;
QAction *actionToggleCheckerboard; QAction *actionToggleCheckerboard;
QAction *actionRotateClockwise; QAction *actionRotateClockwise;
QAction *actionRotateCounterClockwise;
QAction *actionPrevPicture;
QAction *actionPrevPicture; QAction *actionNextPicture;
QAction *actionNextPicture;
QAction *actionHorizontalFlip;
QAction *actionTogglePauseAnimation; QAction *actionFitInView;
QAction *actionAnimationNextFrame; QAction *actionFitByWidth;
QAction *actionCopyPixmap;
QAction *actionHorizontalFlip; QAction *actionCopyFilePath;
QAction *actionFitInView; QAction *actionPaste;
QAction *actionFitByWidth; QAction *actionToggleStayOnTop;
QAction *actionCopyPixmap; QAction *actionToggleProtectMode;
QAction *actionCopyFilePath; QAction *actionToggleAvoidResetTransform;
QAction *actionPaste; QAction *actionSettings;
QAction *actionTrash; QAction *actionHelp;
QAction *actionToggleStayOnTop; QAction *actionLocateInFileManager;
QAction *actionToggleProtectMode; QAction *actionProperties;
QAction *actionToggleAvoidResetTransform; QAction *actionQuitApp;
QAction *actionSettings; };
QAction *actionHelp;
QAction *actionLocateInFileManager; #endif // ACTIONMANAGER_H
QAction *actionProperties;
QAction *actionQuitApp;
};
#endif // ACTIONMANAGER_H

View File

@ -1,57 +1,59 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "bottombuttongroup.h" #include "bottombuttongroup.h"
#include "opacityhelper.h" #include "opacityhelper.h"
#include <QToolButton> #include <functional>
#include <QVBoxLayout>
#include <QDebug> #include <QToolButton>
#include <QVBoxLayout>
BottomButtonGroup::BottomButtonGroup(const std::vector<QAction *> &actionList, QWidget *parent) #include <QDebug>
: QGroupBox (parent)
, m_opacityHelper(new OpacityHelper(this)) BottomButtonGroup::BottomButtonGroup(const std::vector<QAction *> &actionList, QWidget *parent)
{ : QGroupBox (parent)
QHBoxLayout * mainLayout = new QHBoxLayout(this); , m_opacityHelper(new OpacityHelper(this))
mainLayout->setSizeConstraint(QLayout::SetFixedSize); {
this->setLayout(mainLayout); QHBoxLayout * mainLayout = new QHBoxLayout(this);
this->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Minimum); mainLayout->setSizeConstraint(QLayout::SetFixedSize);
this->setStyleSheet("BottomButtonGroup {" this->setLayout(mainLayout);
"border: 1px solid gray;" this->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Minimum);
"border-top-left-radius: 10px;" this->setStyleSheet("BottomButtonGroup {"
"border-top-right-radius: 10px;" "border: 1px solid gray;"
"border-style: none;" "border-top-left-radius: 10px;"
"background-color:rgba(0,0,0,120)" "border-top-right-radius: 10px;"
"}" "border-style: none;"
"QToolButton {" "background-color:rgba(0,0,0,120)"
"background:transparent;" "}"
"}" "QToolButton {"
"QToolButton:!focus {" "background:transparent;"
"border-style: none;" "}"
"}"); "QToolButton:!focus {"
"border-style: none;"
auto newActionBtn = [this](QAction * action) -> QToolButton * { "}");
QToolButton * btn = new QToolButton(this);
btn->setDefaultAction(action); auto newActionBtn = [this](QAction * action) -> QToolButton * {
btn->setIconSize(QSize(32, 32)); QToolButton * btn = new QToolButton(this);
btn->setFixedSize(40, 40); btn->setDefaultAction(action);
return btn; btn->setIconSize(QSize(32, 32));
}; btn->setFixedSize(40, 40);
return btn;
for (QAction * action : actionList) { };
addButton(newActionBtn(action));
} for (QAction * action : actionList) {
} addButton(newActionBtn(action));
}
void BottomButtonGroup::setOpacity(qreal opacity, bool animated) }
{
m_opacityHelper->setOpacity(opacity, animated); void BottomButtonGroup::setOpacity(qreal opacity, bool animated)
} {
m_opacityHelper->setOpacity(opacity, animated);
void BottomButtonGroup::addButton(QAbstractButton *button) }
{
layout()->addWidget(button); void BottomButtonGroup::addButton(QAbstractButton *button)
updateGeometry(); {
} layout()->addWidget(button);
updateGeometry();
}

View File

@ -1,27 +1,27 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#ifndef BOTTOMBUTTONGROUP_H #ifndef BOTTOMBUTTONGROUP_H
#define BOTTOMBUTTONGROUP_H #define BOTTOMBUTTONGROUP_H
#include <vector> #include <vector>
#include <QAbstractButton> #include <QAbstractButton>
#include <QGroupBox> #include <QGroupBox>
class OpacityHelper; class OpacityHelper;
class BottomButtonGroup : public QGroupBox class BottomButtonGroup : public QGroupBox
{ {
Q_OBJECT Q_OBJECT
public: public:
explicit BottomButtonGroup(const std::vector<QAction *> & actionList, QWidget *parent = nullptr); explicit BottomButtonGroup(const std::vector<QAction *> & actionList, QWidget *parent = nullptr);
void setOpacity(qreal opacity, bool animated = true); void setOpacity(qreal opacity, bool animated = true);
void addButton(QAbstractButton *button); void addButton(QAbstractButton *button);
private: private:
OpacityHelper * m_opacityHelper; OpacityHelper * m_opacityHelper;
}; };
#endif // BOTTOMBUTTONGROUP_H #endif // BOTTOMBUTTONGROUP_H

View File

@ -1,138 +1,138 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "exiv2wrapper.h" #include "exiv2wrapper.h"
#ifdef HAVE_EXIV2_VERSION #ifdef HAVE_EXIV2_VERSION
#include <exiv2/exiv2.hpp> #include <exiv2/exiv2.hpp>
#else // HAVE_EXIV2_VERSION #else // HAVE_EXIV2_VERSION
namespace Exiv2 { namespace Exiv2 {
class Image {}; class Image {};
} }
#endif // HAVE_EXIV2_VERSION #endif // HAVE_EXIV2_VERSION
#include <sstream> #include <sstream>
#include <QFile> #include <QFile>
#include <QDebug> #include <QDebug>
Exiv2Wrapper::Exiv2Wrapper() Exiv2Wrapper::Exiv2Wrapper()
{ {
} }
Exiv2Wrapper::~Exiv2Wrapper() Exiv2Wrapper::~Exiv2Wrapper()
{ {
} }
#ifdef HAVE_EXIV2_VERSION // stupid AppleClang... #ifdef HAVE_EXIV2_VERSION // stupid AppleClang...
template<typename Collection, typename Iterator> template<typename Collection, typename Iterator>
void Exiv2Wrapper::cacheSection(Collection collection) void Exiv2Wrapper::cacheSection(Collection collection)
{ {
const Collection& exifData = collection; const Collection& exifData = collection;
Iterator it = exifData.begin(), end = exifData.end(); Iterator it = exifData.begin(), end = exifData.end();
for (; it != end; ++it) { for (; it != end; ++it) {
QString key = QString::fromUtf8(it->key().c_str()); QString key = QString::fromUtf8(it->key().c_str());
if (it->tagName().substr(0, 2) == "0x") continue; if (it->tagName().substr(0, 2) == "0x") continue;
// We might get exceptions like "No namespace info available for XMP prefix `Item'" // 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. // 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... // We don't care for those rare tags so let's just use a try-cache...
try { try {
QString label = QString::fromLocal8Bit(it->tagLabel().c_str()); QString label = QString::fromLocal8Bit(it->tagLabel().c_str());
std::ostringstream stream; std::ostringstream stream;
stream << *it; stream << *it;
QString value = QString::fromUtf8(stream.str().c_str()); QString value = QString::fromUtf8(stream.str().c_str());
m_metadataValue.insert(key, value); m_metadataValue.insert(key, value);
m_metadataLabel.insert(key, label); m_metadataLabel.insert(key, label);
qDebug() << key << label << value; qDebug() << key << label << value;
#if EXIV2_TEST_VERSION(0, 28, 0) #if EXIV2_TEST_VERSION(0, 28, 0)
} catch (Exiv2::Error & err) { } catch (Exiv2::Error & err) {
#else // 0.27.x #else // 0.27.x
} catch (Exiv2::AnyError & err) { } catch (Exiv2::AnyError & err) {
#endif // EXIV2_TEST_VERSION(0, 28, 0) #endif // EXIV2_TEST_VERSION(0, 28, 0)
qWarning() << "Error loading key" << key << ":" << err.what(); qWarning() << "Error loading key" << key << ":" << err.what();
} }
} }
} }
#endif // HAVE_EXIV2_VERSION #endif // HAVE_EXIV2_VERSION
bool Exiv2Wrapper::load(const QString &filePath) bool Exiv2Wrapper::load(const QString &filePath)
{ {
#ifdef HAVE_EXIV2_VERSION #ifdef HAVE_EXIV2_VERSION
QByteArray filePathByteArray = QFile::encodeName(filePath); QByteArray filePathByteArray = QFile::encodeName(filePath);
try { try {
m_exivImage.reset(Exiv2::ImageFactory::open(filePathByteArray.constData()).release()); m_exivImage.reset(Exiv2::ImageFactory::open(filePathByteArray.constData()).release());
m_exivImage->readMetadata(); m_exivImage->readMetadata();
} catch (const Exiv2::Error& error) { } catch (const Exiv2::Error& error) {
m_errMsg = QString::fromUtf8(error.what()); m_errMsg = QString::fromUtf8(error.what());
return false; return false;
} }
return true; return true;
#else // HAVE_EXIV2_VERSION #else // HAVE_EXIV2_VERSION
Q_UNUSED(filePath); Q_UNUSED(filePath);
return false; return false;
#endif // HAVE_EXIV2_VERSION #endif // HAVE_EXIV2_VERSION
} }
void Exiv2Wrapper::cacheSections() void Exiv2Wrapper::cacheSections()
{ {
#ifdef HAVE_EXIV2_VERSION #ifdef HAVE_EXIV2_VERSION
if (m_exivImage->checkMode(Exiv2::mdExif) & Exiv2::amRead) { if (m_exivImage->checkMode(Exiv2::mdExif) & Exiv2::amRead) {
cacheSection<Exiv2::ExifData, Exiv2::ExifData::const_iterator>(m_exivImage->exifData()); cacheSection<Exiv2::ExifData, Exiv2::ExifData::const_iterator>(m_exivImage->exifData());
} }
if (m_exivImage->checkMode(Exiv2::mdIptc) & Exiv2::amRead) { if (m_exivImage->checkMode(Exiv2::mdIptc) & Exiv2::amRead) {
cacheSection<Exiv2::IptcData, Exiv2::IptcData::const_iterator>(m_exivImage->iptcData()); cacheSection<Exiv2::IptcData, Exiv2::IptcData::const_iterator>(m_exivImage->iptcData());
} }
if (m_exivImage->checkMode(Exiv2::mdXmp) & Exiv2::amRead) { if (m_exivImage->checkMode(Exiv2::mdXmp) & Exiv2::amRead) {
cacheSection<Exiv2::XmpData, Exiv2::XmpData::const_iterator>(m_exivImage->xmpData()); cacheSection<Exiv2::XmpData, Exiv2::XmpData::const_iterator>(m_exivImage->xmpData());
} }
// qDebug() << m_metadataValue; // qDebug() << m_metadataValue;
// qDebug() << m_metadataLabel; // qDebug() << m_metadataLabel;
#endif // HAVE_EXIV2_VERSION #endif // HAVE_EXIV2_VERSION
} }
QString Exiv2Wrapper::comment() const QString Exiv2Wrapper::comment() const
{ {
#ifdef HAVE_EXIV2_VERSION #ifdef HAVE_EXIV2_VERSION
return m_exivImage->comment().c_str(); return m_exivImage->comment().c_str();
#else // HAVE_EXIV2_VERSION #else // HAVE_EXIV2_VERSION
return QString(); return QString();
#endif // HAVE_EXIV2_VERSION #endif // HAVE_EXIV2_VERSION
} }
QString Exiv2Wrapper::label(const QString &key) const QString Exiv2Wrapper::label(const QString &key) const
{ {
return m_metadataLabel.value(key); return m_metadataLabel.value(key);
} }
QString Exiv2Wrapper::value(const QString &key) const QString Exiv2Wrapper::value(const QString &key) const
{ {
return m_metadataValue.value(key); return m_metadataValue.value(key);
} }
QString Exiv2Wrapper::XmpValue(const QString &rawValue) QString Exiv2Wrapper::XmpValue(const QString &rawValue)
{ {
QString ignored; QString ignored;
return Exiv2Wrapper::XmpValue(rawValue, ignored); return Exiv2Wrapper::XmpValue(rawValue, ignored);
} }
QString Exiv2Wrapper::XmpValue(const QString &rawValue, QString &language) QString Exiv2Wrapper::XmpValue(const QString &rawValue, QString &language)
{ {
if (rawValue.size() > 6 && rawValue.startsWith(QLatin1String("lang=\""))) { if (rawValue.size() > 6 && rawValue.startsWith(QLatin1String("lang=\""))) {
int pos = rawValue.indexOf('"', 6); int pos = rawValue.indexOf('"', 6);
if (pos != -1) { if (pos != -1) {
language = rawValue.mid(6, pos - 6); language = rawValue.mid(6, pos - 6);
return (rawValue.mid(pos + 2)); return (rawValue.mid(pos + 2));
} }
} }
language.clear(); language.clear();
return rawValue; return rawValue;
} }

View File

@ -1,43 +1,43 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#ifndef EXIV2WRAPPER_H #ifndef EXIV2WRAPPER_H
#define EXIV2WRAPPER_H #define EXIV2WRAPPER_H
#include <memory> #include <memory>
#include <QString> #include <QString>
#include <QMap> #include <QMap>
namespace Exiv2 { namespace Exiv2 {
class Image; class Image;
} }
class Exiv2Wrapper class Exiv2Wrapper
{ {
public: public:
Exiv2Wrapper(); Exiv2Wrapper();
~Exiv2Wrapper(); ~Exiv2Wrapper();
bool load(const QString& filePath); bool load(const QString& filePath);
void cacheSections(); void cacheSections();
QString comment() const; QString comment() const;
QString label(const QString & key) const; QString label(const QString & key) const;
QString value(const QString & key) const; QString value(const QString & key) const;
static QString XmpValue(const QString &rawValue); static QString XmpValue(const QString &rawValue);
static QString XmpValue(const QString &rawValue, QString & language); static QString XmpValue(const QString &rawValue, QString & language);
private: private:
std::unique_ptr<Exiv2::Image> m_exivImage; std::unique_ptr<Exiv2::Image> m_exivImage;
QMap<QString, QString> m_metadataValue; QMap<QString, QString> m_metadataValue;
QMap<QString, QString> m_metadataLabel; QMap<QString, QString> m_metadataLabel;
QString m_errMsg; QString m_errMsg;
template<typename Collection, typename Iterator> template<typename Collection, typename Iterator>
void cacheSection(Collection collection); void cacheSection(Collection collection);
}; };
#endif // EXIV2WRAPPER_H #endif // EXIV2WRAPPER_H

View File

@ -1,22 +0,0 @@
// SPDX-FileCopyrightText: 2024 Gary Wang <git@blumia.net>
//
// SPDX-License-Identifier: MIT
#include "fileopeneventhandler.h"
#include <QFileOpenEvent>
FileOpenEventHandler::FileOpenEventHandler(QObject *parent)
: QObject(parent)
{
}
bool FileOpenEventHandler::eventFilter(QObject *obj, QEvent *event)
{
if (event->type() == QEvent::FileOpen) {
QFileOpenEvent *fileOpenEvent = static_cast<QFileOpenEvent *>(event);
emit fileOpen(fileOpenEvent->url());
return true;
}
return QObject::eventFilter(obj, event);
}

View File

@ -1,21 +0,0 @@
// SPDX-FileCopyrightText: 2024 Gary Wang <git@blumia.net>
//
// SPDX-License-Identifier: MIT
#pragma once
#include <QObject>
class FileOpenEventHandler : public QObject
{
Q_OBJECT
public:
explicit FileOpenEventHandler(QObject *parent = nullptr);
protected:
bool eventFilter(QObject *obj, QEvent *event) override;
signals:
void fileOpen(const QUrl &url);
};

View File

@ -1,136 +1,147 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// SPDX-FileCopyrightText: 2023 Tad Young <yyc12321@outlook.com> // SPDX-FileCopyrightText: 2023 Tad Young <yyc12321@outlook.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "framelesswindow.h" #include "framelesswindow.h"
#include <QMouseEvent> #include <QMouseEvent>
#include <QHoverEvent> #include <QHoverEvent>
#include <QApplication> #include <QApplication>
#include <QVBoxLayout> #include <QVBoxLayout>
#include <QWindow> #include <QWindow>
FramelessWindow::FramelessWindow(QWidget *parent) FramelessWindow::FramelessWindow(QWidget *parent)
: QWidget(parent) : QWidget(parent)
, m_centralLayout(new QVBoxLayout(this)) , m_centralLayout(new QVBoxLayout(this))
, m_oldCursorShape(Qt::ArrowCursor) , m_oldCursorShape(Qt::ArrowCursor)
, m_oldEdges() , m_oldEdges()
{ {
this->setWindowFlags(Qt::Window | Qt::FramelessWindowHint | Qt::WindowMinMaxButtonsHint); #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
this->setMouseTracking(true); this->setWindowFlags(Qt::Window | Qt::FramelessWindowHint | Qt::WindowMinMaxButtonsHint);
this->setAttribute(Qt::WA_Hover, true); #else
this->installEventFilter(this); // 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.
m_centralLayout->setContentsMargins(QMargins()); // 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)
void FramelessWindow::setCentralWidget(QWidget *widget) this->setMouseTracking(true);
{ this->setAttribute(Qt::WA_Hover, true);
if (m_centralWidget) { this->installEventFilter(this);
m_centralLayout->removeWidget(m_centralWidget);
m_centralWidget->deleteLater(); m_centralLayout->setContentsMargins(QMargins());
} }
m_centralLayout->addWidget(widget); void FramelessWindow::setCentralWidget(QWidget *widget)
m_centralWidget = widget; {
} if (m_centralWidget) {
m_centralLayout->removeWidget(m_centralWidget);
void FramelessWindow::installResizeCapture(QObject* widget) m_centralWidget->deleteLater();
{ }
widget->installEventFilter(this);
} m_centralLayout->addWidget(widget);
m_centralWidget = widget;
bool FramelessWindow::eventFilter(QObject* o, QEvent* e) }
{
switch (e->type()) { void FramelessWindow::installResizeCapture(QObject* widget)
case QEvent::HoverMove: {
{ widget->installEventFilter(this);
QWidget* wg = qobject_cast<QWidget*>(o); }
if (wg != nullptr)
return mouseHover(static_cast<QHoverEvent*>(e), wg); bool FramelessWindow::eventFilter(QObject* o, QEvent* e)
{
break; switch (e->type()) {
} case QEvent::HoverMove:
case QEvent::MouseButtonPress: {
return mousePress(static_cast<QMouseEvent*>(e)); QWidget* wg = qobject_cast<QWidget*>(o);
} if (wg != nullptr)
return mouseHover(static_cast<QHoverEvent*>(e), wg);
return QWidget::eventFilter(o, e);
} break;
}
bool FramelessWindow::mouseHover(QHoverEvent* event, QWidget* wg) case QEvent::MouseButtonPress:
{ return mousePress(static_cast<QMouseEvent*>(e));
if (!isMaximized() && !isFullScreen()) { }
QWindow* win = window()->windowHandle();
Qt::Edges edges = this->getEdgesByPos(wg->mapToGlobal(event->oldPos()), win->frameGeometry()); return QWidget::eventFilter(o, e);
}
// backup & restore cursor shape
if (edges && !m_oldEdges) bool FramelessWindow::mouseHover(QHoverEvent* event, QWidget* wg)
// entering the edge. backup cursor shape {
m_oldCursorShape = win->cursor().shape(); if (!isMaximized() && !isFullScreen()) {
if (!edges && m_oldEdges) QWindow* win = window()->windowHandle();
// leaving the edge. restore cursor shape Qt::Edges edges = this->getEdgesByPos(wg->mapToGlobal(event->oldPos()), win->frameGeometry());
win->setCursor(m_oldCursorShape);
// backup & restore cursor shape
// save the latest edges status if (edges && !m_oldEdges)
m_oldEdges = edges; // entering the edge. backup cursor shape
m_oldCursorShape = win->cursor().shape();
// show resize cursor shape if cursor is within border if (!edges && m_oldEdges)
if (edges) { // leaving the edge. restore cursor shape
win->setCursor(this->getCursorByEdge(edges, Qt::ArrowCursor)); win->setCursor(m_oldCursorShape);
return true;
} // save the latest edges status
} m_oldEdges = edges;
return false; // show resize cursor shape if cursor is within border
} if (edges) {
win->setCursor(this->getCursorByEdge(edges, Qt::ArrowCursor));
bool FramelessWindow::mousePress(QMouseEvent* event) return true;
{ }
if (event->buttons() & Qt::LeftButton && !isMaximized() && !isFullScreen()) { }
QWindow* win = window()->windowHandle();
Qt::Edges edges = this->getEdgesByPos(event->globalPosition().toPoint(), win->frameGeometry()); return false;
if (edges) { }
win->startSystemResize(edges);
return true; bool FramelessWindow::mousePress(QMouseEvent* event)
} {
} if (event->buttons() & Qt::LeftButton && !isMaximized() && !isFullScreen()) {
QWindow* win = window()->windowHandle();
return false; #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
} Qt::Edges edges = this->getEdgesByPos(event->globalPosition().toPoint(), win->frameGeometry());
#else
Qt::CursorShape FramelessWindow::getCursorByEdge(const Qt::Edges& edges, Qt::CursorShape default_cursor) Qt::Edges edges = this->getEdgesByPos(event->globalPos(), win->frameGeometry());
{ #endif // QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
if ((edges == (Qt::TopEdge | Qt::LeftEdge)) || (edges == (Qt::RightEdge | Qt::BottomEdge))) if (edges) {
return Qt::SizeFDiagCursor; win->startSystemResize(edges);
else if ((edges == (Qt::TopEdge | Qt::RightEdge)) || (edges == (Qt::LeftEdge | Qt::BottomEdge))) return true;
return Qt::SizeBDiagCursor; }
else if (edges & (Qt::TopEdge | Qt::BottomEdge)) }
return Qt::SizeVerCursor;
else if (edges & (Qt::LeftEdge | Qt::RightEdge)) return false;
return Qt::SizeHorCursor; }
else
return default_cursor; Qt::CursorShape FramelessWindow::getCursorByEdge(const Qt::Edges& edges, Qt::CursorShape default_cursor)
} {
if ((edges == (Qt::TopEdge | Qt::LeftEdge)) || (edges == (Qt::RightEdge | Qt::BottomEdge)))
Qt::Edges FramelessWindow::getEdgesByPos(const QPoint gpos, const QRect& winrect) return Qt::SizeFDiagCursor;
{ else if ((edges == (Qt::TopEdge | Qt::RightEdge)) || (edges == (Qt::LeftEdge | Qt::BottomEdge)))
const int borderWidth = 8; return Qt::SizeBDiagCursor;
Qt::Edges edges; else if (edges & (Qt::TopEdge | Qt::BottomEdge))
return Qt::SizeVerCursor;
int x = gpos.x() - winrect.x(); else if (edges & (Qt::LeftEdge | Qt::RightEdge))
int y = gpos.y() - winrect.y(); return Qt::SizeHorCursor;
else
if (x < borderWidth) return default_cursor;
edges |= Qt::LeftEdge; }
if (x > (winrect.width() - borderWidth))
edges |= Qt::RightEdge; Qt::Edges FramelessWindow::getEdgesByPos(const QPoint gpos, const QRect& winrect)
if (y < borderWidth) {
edges |= Qt::TopEdge; const int borderWidth = 8;
if (y > (winrect.height() - borderWidth)) Qt::Edges edges;
edges |= Qt::BottomEdge;
int x = gpos.x() - winrect.x();
return edges; 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;
}

View File

@ -1,39 +1,39 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#ifndef FRAMELESSWINDOW_H #ifndef FRAMELESSWINDOW_H
#define FRAMELESSWINDOW_H #define FRAMELESSWINDOW_H
#include <QWidget> #include <QWidget>
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE
class QVBoxLayout; class QVBoxLayout;
QT_END_NAMESPACE QT_END_NAMESPACE
class FramelessWindow : public QWidget class FramelessWindow : public QWidget
{ {
Q_OBJECT Q_OBJECT
public: public:
explicit FramelessWindow(QWidget *parent = nullptr); explicit FramelessWindow(QWidget *parent = nullptr);
void setCentralWidget(QWidget * widget); void setCentralWidget(QWidget * widget);
void installResizeCapture(QObject* widget); void installResizeCapture(QObject* widget);
protected: protected:
bool eventFilter(QObject *o, QEvent *e) override; bool eventFilter(QObject *o, QEvent *e) override;
bool mouseHover(QHoverEvent* event, QWidget* wg); bool mouseHover(QHoverEvent* event, QWidget* wg);
bool mousePress(QMouseEvent* event); bool mousePress(QMouseEvent* event);
private: private:
Qt::Edges m_oldEdges; Qt::Edges m_oldEdges;
Qt::CursorShape m_oldCursorShape; Qt::CursorShape m_oldCursorShape;
Qt::CursorShape getCursorByEdge(const Qt::Edges& edges, Qt::CursorShape default_cursor); Qt::CursorShape getCursorByEdge(const Qt::Edges& edges, Qt::CursorShape default_cursor);
Qt::Edges getEdgesByPos(const QPoint pos, const QRect& winrect); Qt::Edges getEdgesByPos(const QPoint pos, const QRect& winrect);
QVBoxLayout * m_centralLayout = nullptr; QVBoxLayout * m_centralLayout = nullptr;
QWidget * m_centralWidget = nullptr; // just a pointer, doesn't take the ownership. QWidget * m_centralWidget = nullptr; // just a pointer, doesn't take the ownership.
}; };
#endif // FRAMELESSWINDOW_H #endif // FRAMELESSWINDOW_H

View File

@ -10,7 +10,6 @@
#include <QGraphicsItem> #include <QGraphicsItem>
#include <QUrl> #include <QUrl>
#include <QGraphicsSvgItem> #include <QGraphicsSvgItem>
#include <QSvgRenderer>
#include <QMovie> #include <QMovie>
#include <QPainter> #include <QPainter>
@ -21,9 +20,6 @@ public:
: QGraphicsPixmapItem(pixmap, parent) : QGraphicsPixmapItem(pixmap, parent)
{} {}
enum { Type = UserType + 1 };
int type() const override { return Type; }
void setScaleHint(float scaleHint) { void setScaleHint(float scaleHint) {
m_scaleHint = scaleHint; m_scaleHint = scaleHint;
} }
@ -64,9 +60,6 @@ class PGraphicsMovieItem : public QGraphicsItem
public: public:
PGraphicsMovieItem(QGraphicsItem *parent = nullptr) : QGraphicsItem(parent) {} PGraphicsMovieItem(QGraphicsItem *parent = nullptr) : QGraphicsItem(parent) {}
enum { Type = UserType + 2 };
int type() const override { return Type; }
void setMovie(QMovie* movie) { void setMovie(QMovie* movie) {
if (m_movie) m_movie->disconnect(); if (m_movie) m_movie->disconnect();
m_movie.reset(movie); m_movie.reset(movie);
@ -86,10 +79,6 @@ public:
} }
} }
inline QMovie * movie() const {
return m_movie.data();
}
private: private:
QScopedPointer<QMovie> m_movie; QScopedPointer<QMovie> m_movie;
}; };
@ -127,16 +116,7 @@ void GraphicsScene::showText(const QString &text)
void GraphicsScene::showSvg(const QString &filepath) void GraphicsScene::showSvg(const QString &filepath)
{ {
this->clear(); this->clear();
QGraphicsSvgItem * svgItem = new QGraphicsSvgItem(); QGraphicsSvgItem * svgItem = new QGraphicsSvgItem(filepath);
QSvgRenderer * render = new QSvgRenderer(svgItem);
#if QT_VERSION >= QT_VERSION_CHECK(6, 7, 0)
// Qt 6.7.0's SVG support is terrible caused by huge memory usage, see QTBUG-124287
// Qt 6.7.1's is somewhat better, memory issue seems fixed, but still laggy when zoom in,
// see QTBUG-126771. Anyway let's disable it for now.
render->setOptions(QtSvg::Tiny12FeaturesOnly);
#endif // QT_VERSION >= QT_VERSION_CHECK(6, 7, 0)
render->load(filepath);
svgItem->setSharedRenderer(render);
this->addItem(svgItem); this->addItem(svgItem);
m_theThing = svgItem; m_theThing = svgItem;
this->setSceneRect(m_theThing->boundingRect()); this->setSceneRect(m_theThing->boundingRect());
@ -168,29 +148,6 @@ bool GraphicsScene::trySetTransformationMode(Qt::TransformationMode mode, float
return false; return false;
} }
bool GraphicsScene::togglePauseAnimation()
{
PGraphicsMovieItem * animatedItem = qgraphicsitem_cast<PGraphicsMovieItem *>(m_theThing);
if (animatedItem) {
animatedItem->movie()->setPaused(animatedItem->movie()->state() != QMovie::Paused);
return true;
}
return false;
}
bool GraphicsScene::skipAnimationFrame(int delta)
{
PGraphicsMovieItem * animatedItem = qgraphicsitem_cast<PGraphicsMovieItem *>(m_theThing);
if (animatedItem) {
const int frameCount = animatedItem->movie()->frameCount();
const int currentFrame = animatedItem->movie()->currentFrameNumber();
const int targetFrame = (currentFrame + delta) % frameCount;
animatedItem->movie()->setPaused(true);
return animatedItem->movie()->jumpToFrame(targetFrame);
}
return false;
}
QPixmap GraphicsScene::renderToPixmap() QPixmap GraphicsScene::renderToPixmap()
{ {
PGraphicsPixmapItem * pixmapItem = qgraphicsitem_cast<PGraphicsPixmapItem *>(m_theThing); PGraphicsPixmapItem * pixmapItem = qgraphicsitem_cast<PGraphicsPixmapItem *>(m_theThing);

View File

@ -21,13 +21,10 @@ public:
bool trySetTransformationMode(Qt::TransformationMode mode, float scaleHint); bool trySetTransformationMode(Qt::TransformationMode mode, float scaleHint);
bool togglePauseAnimation();
bool skipAnimationFrame(int delta = 1);
QPixmap renderToPixmap(); QPixmap renderToPixmap();
private: private:
QGraphicsItem * m_theThing = nullptr; QGraphicsItem * m_theThing;
}; };
#endif // GRAPHICSSCENE_H #endif // GRAPHICSSCENE_H

View File

@ -1,440 +1,422 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "graphicsview.h" #include "graphicsview.h"
#include "graphicsscene.h" #include "graphicsscene.h"
#include "settings.h"
#include <QDebug>
#include <QDebug> #include <QMouseEvent>
#include <QMouseEvent> #include <QScrollBar>
#include <QScrollBar> #include <QMimeData>
#include <QMimeData> #include <QImageReader>
#include <QImageReader> #include <QStyleOptionGraphicsItem>
#include <QStyleOptionGraphicsItem>
GraphicsView::GraphicsView(QWidget *parent)
GraphicsView::GraphicsView(QWidget *parent) : QGraphicsView (parent)
: QGraphicsView (parent) {
{ setDragMode(QGraphicsView::ScrollHandDrag);
setDragMode(QGraphicsView::ScrollHandDrag); setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setResizeAnchor(QGraphicsView::AnchorUnderMouse);
setResizeAnchor(QGraphicsView::AnchorUnderMouse); setTransformationAnchor(QGraphicsView::AnchorUnderMouse);
setTransformationAnchor(QGraphicsView::AnchorUnderMouse); setStyleSheet("background-color: rgba(0, 0, 0, 220);"
setStyleSheet("background-color: rgba(0, 0, 0, 220);" "border-radius: 3px;");
"border-radius: 3px;"); setAcceptDrops(true);
setAcceptDrops(false); setCheckerboardEnabled(false);
setCheckerboardEnabled(false);
connect(horizontalScrollBar(), &QScrollBar::valueChanged, this, &GraphicsView::viewportRectChanged);
connect(horizontalScrollBar(), &QScrollBar::valueChanged, this, &GraphicsView::viewportRectChanged); connect(verticalScrollBar(), &QScrollBar::valueChanged, this, &GraphicsView::viewportRectChanged);
connect(verticalScrollBar(), &QScrollBar::valueChanged, this, &GraphicsView::viewportRectChanged); }
}
void GraphicsView::showFileFromPath(const QString &filePath, bool doRequestGallery)
void GraphicsView::showFileFromPath(const QString &filePath) {
{ emit navigatorViewRequired(false, transform());
emit navigatorViewRequired(false, transform());
if (filePath.endsWith(".svg")) {
if (filePath.endsWith(".svg")) { showSvg(filePath);
showSvg(filePath); } else {
} else { QImageReader imageReader(filePath);
QImageReader imageReader(filePath); imageReader.setAutoTransform(true);
imageReader.setAutoTransform(true); imageReader.setDecideFormatFromContent(true);
imageReader.setDecideFormatFromContent(true); #if QT_VERSION > QT_VERSION_CHECK(6, 0, 0)
imageReader.setAllocationLimit(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. // Since if the image format / plugin does not support this feature, imageFormat() will returns an invalid format.
// QImage::Format imageFormat = imageReader.imageFormat(); // So we cannot use imageFormat() and check if it returns QImage::Format_Invalid to detect if we support the file.
if (imageReader.format().isEmpty()) { // QImage::Format imageFormat = imageReader.imageFormat();
showText(tr("File is not a valid image")); if (imageReader.format().isEmpty()) {
} else if (imageReader.supportsAnimation() && imageReader.imageCount() > 1) { doRequestGallery = false;
showAnimated(filePath); showText(tr("File is not a valid image"));
} else if (!imageReader.canRead()) { } else if (imageReader.supportsAnimation() && imageReader.imageCount() > 1) {
showText(tr("Image data is invalid or currently unsupported")); showAnimated(filePath);
} else { } else if (!imageReader.canRead()) {
QPixmap && pixmap = QPixmap::fromImageReader(&imageReader); doRequestGallery = false;
if (pixmap.isNull()) { showText(tr("Image data is invalid or currently unsupported"));
showText(tr("Image data is invalid or currently unsupported")); } else {
} else { QPixmap && pixmap = QPixmap::fromImageReader(&imageReader);
pixmap.setDevicePixelRatio(devicePixelRatioF()); if (pixmap.isNull()) {
showImage(pixmap); doRequestGallery = false;
} showText(tr("Image data is invalid or currently unsupported"));
} } else {
} pixmap.setDevicePixelRatio(devicePixelRatioF());
} showImage(pixmap);
}
void GraphicsView::showImage(const QPixmap &pixmap) }
{ }
resetTransform();
scene()->showImage(pixmap); if (doRequestGallery) {
displayScene(); emit requestGallery(filePath);
} }
}
void GraphicsView::showImage(const QImage &image)
{ void GraphicsView::showImage(const QPixmap &pixmap)
resetTransform(); {
scene()->showImage(QPixmap::fromImage(image)); resetTransform();
displayScene(); scene()->showImage(pixmap);
} displayScene();
}
void GraphicsView::showText(const QString &text)
{ void GraphicsView::showImage(const QImage &image)
resetTransform(); {
scene()->showText(text); resetTransform();
displayScene(); scene()->showImage(QPixmap::fromImage(image));
} displayScene();
}
void GraphicsView::showSvg(const QString &filepath)
{ void GraphicsView::showText(const QString &text)
resetTransform(); {
scene()->showSvg(filepath); resetTransform();
displayScene(); scene()->showText(text);
} displayScene();
}
void GraphicsView::showAnimated(const QString &filepath)
{ void GraphicsView::showSvg(const QString &filepath)
resetTransform(); {
scene()->showAnimated(filepath); resetTransform();
displayScene(); scene()->showSvg(filepath);
} displayScene();
}
GraphicsScene *GraphicsView::scene() const
{ void GraphicsView::showAnimated(const QString &filepath)
return qobject_cast<GraphicsScene*>(QGraphicsView::scene()); {
} resetTransform();
scene()->showAnimated(filepath);
void GraphicsView::setScene(GraphicsScene *scene) displayScene();
{ }
return QGraphicsView::setScene(scene);
} GraphicsScene *GraphicsView::scene() const
{
qreal GraphicsView::scaleFactor() const return qobject_cast<GraphicsScene*>(QGraphicsView::scene());
{ }
return QStyleOptionGraphicsItem::levelOfDetailFromTransform(transform());
} void GraphicsView::setScene(GraphicsScene *scene)
{
void GraphicsView::resetTransform() return QGraphicsView::setScene(scene);
{ }
if (!shouldAvoidTransform()) {
QGraphicsView::resetTransform(); qreal GraphicsView::scaleFactor() const
} {
} return QStyleOptionGraphicsItem::levelOfDetailFromTransform(transform());
}
void GraphicsView::zoomView(qreal scaleFactor)
{ void GraphicsView::resetTransform()
m_enableFitInView = false; {
scale(scaleFactor, scaleFactor); if (!m_avoidResetTransform) {
applyTransformationModeByScaleFactor(); QGraphicsView::resetTransform();
emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform()); }
} }
// This is always according to user's view. void GraphicsView::zoomView(qreal scaleFactor)
// the direction of the rotation will NOT be clockwise because the y-axis points downwards. {
void GraphicsView::rotateView(bool clockwise) m_enableFitInView = false;
{ scale(scaleFactor, scaleFactor);
resetScale(); applyTransformationModeByScaleFactor();
emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform());
QTransform tf(0, clockwise ? 1 : -1, 0, }
clockwise ? -1 : 1, 0, 0,
0, 0, 1); // This is always according to user's view.
tf = transform() * tf; // the direction of the rotation will NOT be clockwise because the y-axis points downwards.
setTransform(tf); void GraphicsView::rotateView(bool clockwise)
} {
resetScale();
void GraphicsView::flipView(bool horizontal)
{ QTransform tf(0, clockwise ? 1 : -1, 0,
QTransform tf(horizontal ? -1 : 1, 0, 0, clockwise ? -1 : 1, 0, 0,
0, horizontal ? 1 : -1, 0, 0, 0, 1);
0, 0, 1); tf = transform() * tf;
tf = transform() * tf; setTransform(tf);
setTransform(tf); }
// Ensure the navigation view is also flipped. void GraphicsView::flipView(bool horizontal)
emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform()); {
} QTransform tf(horizontal ? -1 : 1, 0, 0,
0, horizontal ? 1 : -1, 0,
void GraphicsView::resetScale() 0, 0, 1);
{ tf = transform() * tf;
setTransform(resetScale(transform())); setTransform(tf);
applyTransformationModeByScaleFactor();
emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform()); // Ensure the navigation view is also flipped.
} emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform());
}
void GraphicsView::fitInView(const QRectF &rect, Qt::AspectRatioMode aspectRadioMode)
{ void GraphicsView::resetScale()
QGraphicsView::fitInView(rect, aspectRadioMode); {
applyTransformationModeByScaleFactor(); setTransform(resetScale(transform()));
} applyTransformationModeByScaleFactor();
emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform());
void GraphicsView::fitByOrientation(Qt::Orientation ori, bool scaleDownOnly) }
{
resetScale(); void GraphicsView::fitInView(const QRectF &rect, Qt::AspectRatioMode aspectRadioMode)
{
QRectF viewRect = this->viewport()->rect(); QGraphicsView::fitInView(rect, aspectRadioMode);
QRectF imageRect = transform().mapRect(sceneRect()); applyTransformationModeByScaleFactor();
QSize viewSize = viewRect.size().toSize(); }
qreal ratio; void GraphicsView::fitByOrientation(Qt::Orientation ori, bool scaleDownOnly)
{
if (ori == Qt::Horizontal) { resetScale();
// Horizontal fit means fit by width
if (scaleDownOnly && imageRect.width() <= viewSize.width()) { QRectF viewRect = this->viewport()->rect().adjusted(2, 2, -2, -2);
// Image width already fits, no scaling needed QRectF imageRect = transform().mapRect(sceneRect());
ratio = 1;
} else { qreal ratio;
ratio = viewRect.width() / imageRect.width();
} if (ori == Qt::Horizontal) {
} else { ratio = viewRect.width() / imageRect.width();
// Vertical fit means fit by height } else {
if (scaleDownOnly && imageRect.height() <= viewSize.height()) { ratio = viewRect.height() / imageRect.height();
// Image height already fits, no scaling needed }
ratio = 1;
} else { if (scaleDownOnly && ratio > 1) ratio = 1;
ratio = viewRect.height() / imageRect.height();
} scale(ratio, ratio);
} centerOn(imageRect.top(), 0);
m_enableFitInView = false;
if (ratio != 1) {
scale(ratio, ratio); applyTransformationModeByScaleFactor();
} emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform());
}
// Position the image correctly based on orientation with rotation consideration
QRectF originalScene = sceneRect(); void GraphicsView::displayScene()
QTransform currentTransform = transform(); {
if (m_avoidResetTransform) {
if (ori == Qt::Horizontal) { emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform());
// For horizontal fit (fit by width), position at top (for tall images) return;
// Find the scene point that corresponds to the top-center of the transformed image }
QPointF sceneTopCenter;
if (isSceneBiggerThanView()) {
if (qFuzzyIsNull(currentTransform.m12()) && qFuzzyIsNull(currentTransform.m21())) { fitInView(sceneRect(), Qt::KeepAspectRatio);
// 0° or 180° rotation }
if (currentTransform.m11() > 0 && currentTransform.m22() > 0) {
// 0° rotation: use original top-center m_enableFitInView = true;
sceneTopCenter = QPointF(originalScene.center().x(), originalScene.top()); }
} else {
// 180° rotation: the visual "top" is now at the scene bottom bool GraphicsView::isSceneBiggerThanView() const
sceneTopCenter = QPointF(originalScene.center().x(), originalScene.bottom()); {
} if (!isThingSmallerThanWindowWith(transform())) {
} else { return true;
// 90/270 degree rotation: the "top" in view corresponds to left/right in scene } else {
if (currentTransform.m12() > 0) { return false;
// 90 degree: top in view = left in scene }
sceneTopCenter = QPointF(originalScene.left(), originalScene.center().y()); }
} else {
// 270 degree: top in view = right in scene // Automately do fit in view when viewport(window) smaller than image original size.
sceneTopCenter = QPointF(originalScene.right(), originalScene.center().y()); void GraphicsView::setEnableAutoFitInView(bool enable)
} {
} m_enableFitInView = enable;
centerOn(sceneTopCenter); }
} else {
// For vertical fit (fit by height), position at left (for wide images) bool GraphicsView::avoidResetTransform() const
// Find the scene point that corresponds to the left-center of the transformed image {
QPointF sceneLeftCenter; return m_avoidResetTransform;
}
if (qFuzzyIsNull(currentTransform.m12()) && qFuzzyIsNull(currentTransform.m21())) {
// 0° or 180° rotation void GraphicsView::setAvoidResetTransform(bool avoidReset)
if (currentTransform.m11() > 0 && currentTransform.m22() > 0) { {
// 0° rotation: use original left-center m_avoidResetTransform = avoidReset;
sceneLeftCenter = QPointF(originalScene.left(), originalScene.center().y()); }
} else {
// 180° rotation: the visual "left" is now at the scene right inline double zeroOrOne(double number)
sceneLeftCenter = QPointF(originalScene.right(), originalScene.center().y()); {
} return qFuzzyIsNull(number) ? 0 : (number > 0 ? 1 : -1);
} else { }
// 90/270 degree rotation: the "left" in view corresponds to top/bottom in scene
if (currentTransform.m21() > 0) { // Note: this only works if we only have 90 degree based rotation
// 90 degree: left in view = top in scene // and no shear/translate.
sceneLeftCenter = QPointF(originalScene.center().x(), originalScene.top()); QTransform GraphicsView::resetScale(const QTransform & orig)
} else { {
// 270 degree: left in view = bottom in scene return QTransform(zeroOrOne(orig.m11()), zeroOrOne(orig.m12()),
sceneLeftCenter = QPointF(originalScene.center().x(), originalScene.bottom()); zeroOrOne(orig.m21()), zeroOrOne(orig.m22()),
} orig.dx(), orig.dy());
} }
centerOn(sceneLeftCenter);
} void GraphicsView::toggleCheckerboard(bool invertCheckerboardColor)
{
m_enableFitInView = false; setCheckerboardEnabled(!m_checkerboardEnabled, invertCheckerboardColor);
}
applyTransformationModeByScaleFactor();
emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform()); void GraphicsView::mousePressEvent(QMouseEvent *event)
} {
if (shouldIgnoreMousePressMoveEvent(event)) {
void GraphicsView::displayScene() event->ignore();
{ // blumia: return here, or the QMouseEvent event transparency won't
if (shouldAvoidTransform()) { // work if we set a QGraphicsView::ScrollHandDrag drag mode.
emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform()); return;
return; }
}
return QGraphicsView::mousePressEvent(event);
if (isSceneBiggerThanView()) { }
fitInView(sceneRect(), Qt::KeepAspectRatio);
} void GraphicsView::mouseMoveEvent(QMouseEvent *event)
{
m_enableFitInView = true; if (shouldIgnoreMousePressMoveEvent(event)) {
m_firstUserMediaLoaded = true; event->ignore();
} }
bool GraphicsView::isSceneBiggerThanView() const return QGraphicsView::mouseMoveEvent(event);
{ }
if (!isThingSmallerThanWindowWith(transform())) {
return true; void GraphicsView::mouseReleaseEvent(QMouseEvent *event)
} else { {
return false; if (event->button() == Qt::ForwardButton || event->button() == Qt::BackButton) {
} event->ignore();
} } else {
QGraphicsItem *item = itemAt(event->pos());
// Automately do fit in view when viewport(window) smaller than image original size. if (!item) {
void GraphicsView::setEnableAutoFitInView(bool enable) event->ignore();
{ }
m_enableFitInView = enable; }
}
return QGraphicsView::mouseReleaseEvent(event);
bool GraphicsView::avoidResetTransform() const }
{
return m_avoidResetTransform; void GraphicsView::wheelEvent(QWheelEvent *event)
} {
event->ignore();
void GraphicsView::setAvoidResetTransform(bool avoidReset) // blumia: no need for calling parent method.
{ }
m_avoidResetTransform = avoidReset;
} void GraphicsView::resizeEvent(QResizeEvent *event)
{
inline double zeroOrOne(double number) if (m_enableFitInView) {
{ bool originalSizeSmallerThanWindow = isThingSmallerThanWindowWith(resetScale(transform()));
return qFuzzyIsNull(number) ? 0 : (number > 0 ? 1 : -1); if (originalSizeSmallerThanWindow && scaleFactor() >= 1) {
} // no longer need to do fitInView()
// but we leave the m_enableFitInView value unchanged in case
// Note: this only works if we only have 90 degree based rotation // user resize down the window again.
// and no shear/translate. } else if (originalSizeSmallerThanWindow && scaleFactor() < 1) {
QTransform GraphicsView::resetScale(const QTransform & orig) resetScale();
{ } else {
return QTransform(zeroOrOne(orig.m11()), zeroOrOne(orig.m12()), fitInView(sceneRect(), Qt::KeepAspectRatio);
zeroOrOne(orig.m21()), zeroOrOne(orig.m22()), }
orig.dx(), orig.dy()); } else {
} emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform());
}
void GraphicsView::toggleCheckerboard(bool invertCheckerboardColor) return QGraphicsView::resizeEvent(event);
{ }
setCheckerboardEnabled(!m_checkerboardEnabled, invertCheckerboardColor);
} void GraphicsView::dragEnterEvent(QDragEnterEvent *event)
{
void GraphicsView::mousePressEvent(QMouseEvent *event) if (event->mimeData()->hasUrls() || event->mimeData()->hasImage() || event->mimeData()->hasText()) {
{ event->acceptProposedAction();
if (shouldIgnoreMousePressMoveEvent(event)) { } else {
event->ignore(); event->ignore();
// blumia: return here, or the QMouseEvent event transparency won't }
// work if we set a QGraphicsView::ScrollHandDrag drag mode. // qDebug() << event->mimeData() << "Drag Enter Event"
return; // << event->mimeData()->hasUrls() << event->mimeData()->hasImage()
} // << event->mimeData()->formats() << event->mimeData()->hasFormat("text/uri-list");
return QGraphicsView::mousePressEvent(event); return QGraphicsView::dragEnterEvent(event);
} }
void GraphicsView::mouseMoveEvent(QMouseEvent *event) void GraphicsView::dragMoveEvent(QDragMoveEvent *event)
{ {
if (shouldIgnoreMousePressMoveEvent(event)) { Q_UNUSED(event)
event->ignore(); // 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.
}
return QGraphicsView::mouseMoveEvent(event);
} void GraphicsView::dropEvent(QDropEvent *event)
{
void GraphicsView::mouseReleaseEvent(QMouseEvent *event) event->acceptProposedAction();
{
if (event->button() == Qt::ForwardButton || event->button() == Qt::BackButton) { const QMimeData * mimeData = event->mimeData();
event->ignore();
} else { if (mimeData->hasUrls()) {
QGraphicsItem *item = itemAt(event->pos()); const QList<QUrl> &urls = mimeData->urls();
if (!item) { if (urls.isEmpty()) {
event->ignore(); showText(tr("File url list is empty"));
} } else {
} showFileFromPath(urls.first().toLocalFile(), true);
}
return QGraphicsView::mouseReleaseEvent(event); } else if (mimeData->hasImage()) {
} QImage img = qvariant_cast<QImage>(mimeData->imageData());
QPixmap pixmap = QPixmap::fromImage(img);
void GraphicsView::wheelEvent(QWheelEvent *event) if (pixmap.isNull()) {
{ showText(tr("Image data is invalid"));
event->ignore(); } else {
// blumia: no need for calling parent method. showImage(pixmap);
} }
} else if (mimeData->hasText()) {
void GraphicsView::resizeEvent(QResizeEvent *event) showText(mimeData->text());
{ } else {
if (m_enableFitInView) { showText(tr("Not supported mimedata: %1").arg(mimeData->formats().first()));
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 bool GraphicsView::isThingSmallerThanWindowWith(const QTransform &transform) const
// user resize down the window again. {
} else if (originalSizeSmallerThanWindow && scaleFactor() < 1) { return rect().size().expandedTo(transform.mapRect(sceneRect()).size().toSize())
resetScale(); == rect().size();
} else { }
fitInView(sceneRect(), Qt::KeepAspectRatio);
} bool GraphicsView::shouldIgnoreMousePressMoveEvent(const QMouseEvent *event) const
} else { {
emit navigatorViewRequired(!isThingSmallerThanWindowWith(transform()), transform()); if (event->buttons() == Qt::NoButton) {
} return true;
return QGraphicsView::resizeEvent(event); }
}
QGraphicsItem *item = itemAt(event->pos());
bool GraphicsView::isThingSmallerThanWindowWith(const QTransform &transform) const if (!item) {
{ return true;
return rect().size().expandedTo(transform.mapRect(sceneRect()).size().toSize()) }
== rect().size();
} if (isThingSmallerThanWindowWith(transform())) {
return true;
bool GraphicsView::shouldIgnoreMousePressMoveEvent(const QMouseEvent *event) const }
{
if (event->buttons() == Qt::NoButton) { return false;
return true; }
}
void GraphicsView::setCheckerboardEnabled(bool enabled, bool invertColor)
QGraphicsItem *item = itemAt(event->pos()); {
if (!item) { m_checkerboardEnabled = enabled;
return true; m_isLastCheckerboardColorInverted = invertColor;
} if (m_checkerboardEnabled) {
// Prepare background check-board pattern
if (isThingSmallerThanWindowWith(transform())) { QPixmap tilePixmap(0x20, 0x20);
return true; tilePixmap.fill(invertColor ? QColor(220, 220, 220, 170) : QColor(35, 35, 35, 170));
} QPainter tilePainter(&tilePixmap);
constexpr QColor color(45, 45, 45, 170);
return false; 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);
void GraphicsView::setCheckerboardEnabled(bool enabled, bool invertColor) tilePainter.end();
{
m_checkerboardEnabled = enabled; setBackgroundBrush(tilePixmap);
bool isLightCheckerboard = Settings::instance()->useLightCheckerboard() ^ invertColor; } else {
if (m_checkerboardEnabled) { setBackgroundBrush(Qt::transparent);
// Prepare background check-board pattern }
QPixmap tilePixmap(0x20, 0x20); }
tilePixmap.fill(isLightCheckerboard ? QColor(220, 220, 220, 170) : QColor(35, 35, 35, 170));
QPainter tilePainter(&tilePixmap); void GraphicsView::applyTransformationModeByScaleFactor()
constexpr QColor color(45, 45, 45, 170); {
constexpr QColor invertedColor(210, 210, 210, 170); if (this->scaleFactor() < 1) {
tilePainter.fillRect(0, 0, 0x10, 0x10, isLightCheckerboard ? invertedColor : color); scene()->trySetTransformationMode(Qt::SmoothTransformation, this->scaleFactor());
tilePainter.fillRect(0x10, 0x10, 0x10, 0x10, isLightCheckerboard ? invertedColor : color); } else {
tilePainter.end(); scene()->trySetTransformationMode(Qt::FastTransformation, this->scaleFactor());
}
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());
}
}
bool GraphicsView::shouldAvoidTransform() const
{
return m_firstUserMediaLoaded && m_avoidResetTransform;
}

View File

@ -1,79 +1,81 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#ifndef GRAPHICSVIEW_H #ifndef GRAPHICSVIEW_H
#define GRAPHICSVIEW_H #define GRAPHICSVIEW_H
#include <QGraphicsView> #include <QGraphicsView>
#include <QUrl> #include <QUrl>
class GraphicsScene; class GraphicsScene;
class GraphicsView : public QGraphicsView class GraphicsView : public QGraphicsView
{ {
Q_OBJECT Q_OBJECT
public: public:
GraphicsView(QWidget *parent = nullptr); GraphicsView(QWidget *parent = nullptr);
void showFileFromPath(const QString &filePath); void showFileFromPath(const QString &filePath, bool requestGallery = false);
void showImage(const QPixmap &pixmap); void showImage(const QPixmap &pixmap);
void showImage(const QImage &image); void showImage(const QImage &image);
void showText(const QString &text); void showText(const QString &text);
void showSvg(const QString &filepath); void showSvg(const QString &filepath);
void showAnimated(const QString &filepath); void showAnimated(const QString &filepath);
GraphicsScene * scene() const; GraphicsScene * scene() const;
void setScene(GraphicsScene *scene); void setScene(GraphicsScene *scene);
qreal scaleFactor() const; qreal scaleFactor() const;
void resetTransform(); void resetTransform();
void zoomView(qreal scaleFactor); void zoomView(qreal scaleFactor);
void rotateView(bool clockwise = true); void rotateView(bool clockwise = true);
void flipView(bool horizontal = true); void flipView(bool horizontal = true);
void resetScale(); void resetScale();
void fitInView(const QRectF &rect, Qt::AspectRatioMode aspectRadioMode = Qt::IgnoreAspectRatio); void fitInView(const QRectF &rect, Qt::AspectRatioMode aspectRadioMode = Qt::IgnoreAspectRatio);
void fitByOrientation(Qt::Orientation ori = Qt::Horizontal, bool scaleDownOnly = false); void fitByOrientation(Qt::Orientation ori = Qt::Horizontal, bool scaleDownOnly = false);
void displayScene(); void displayScene();
bool isSceneBiggerThanView() const; bool isSceneBiggerThanView() const;
void setEnableAutoFitInView(bool enable = true); void setEnableAutoFitInView(bool enable = true);
bool avoidResetTransform() const; bool avoidResetTransform() const;
void setAvoidResetTransform(bool avoidReset); void setAvoidResetTransform(bool avoidReset);
static QTransform resetScale(const QTransform & orig); static QTransform resetScale(const QTransform & orig);
signals: signals:
void navigatorViewRequired(bool required, QTransform transform); void navigatorViewRequired(bool required, QTransform transform);
void viewportRectChanged(); void viewportRectChanged();
void requestGallery(const QString &filePath);
public slots:
void toggleCheckerboard(bool invertCheckerboardColor = false); public slots:
void toggleCheckerboard(bool invertCheckerboardColor = false);
private:
void mousePressEvent(QMouseEvent * event) override; private:
void mouseMoveEvent(QMouseEvent * event) override; void mousePressEvent(QMouseEvent * event) override;
void mouseReleaseEvent(QMouseEvent * event) override; void mouseMoveEvent(QMouseEvent * event) override;
void wheelEvent(QWheelEvent *event) override; void mouseReleaseEvent(QMouseEvent * event) override;
void resizeEvent(QResizeEvent *event) override; void wheelEvent(QWheelEvent *event) override;
void resizeEvent(QResizeEvent *event) override;
bool isThingSmallerThanWindowWith(const QTransform &transform) const;
bool shouldIgnoreMousePressMoveEvent(const QMouseEvent *event) const; void dragEnterEvent(QDragEnterEvent *event) override;
void setCheckerboardEnabled(bool enabled, bool invertColor = false); void dragMoveEvent(QDragMoveEvent *event) override;
void applyTransformationModeByScaleFactor(); void dropEvent(QDropEvent *event) override;
inline bool shouldAvoidTransform() const; bool isThingSmallerThanWindowWith(const QTransform &transform) const;
bool shouldIgnoreMousePressMoveEvent(const QMouseEvent *event) const;
// Consider switch to 3 state for "no fit", "always fit" and "fit when view is smaller"? void setCheckerboardEnabled(bool enabled, bool invertColor = false);
// ... or even more? e.g. "fit/snap width" things... void applyTransformationModeByScaleFactor();
// Currently it's "no fit" when it's false and "fit when view is smaller" when it's true.
bool m_enableFitInView = false; // Consider switch to 3 state for "no fit", "always fit" and "fit when view is smaller"?
bool m_avoidResetTransform = false; // ... or even more? e.g. "fit/snap width" things...
bool m_checkerboardEnabled = false; // Currently it's "no fit" when it's false and "fit when view is smaller" when it's true.
bool m_useLightCheckerboard = false; bool m_enableFitInView = false;
bool m_firstUserMediaLoaded = false; bool m_avoidResetTransform = false;
}; bool m_checkerboardEnabled = false;
bool m_isLastCheckerboardColorInverted = false;
#endif // GRAPHICSVIEW_H };
#endif // GRAPHICSVIEW_H

View File

@ -1,103 +1,63 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "mainwindow.h" #include "mainwindow.h"
#include "playlistmanager.h" #include "playlistmanager.h"
#include "settings.h" #include "settings.h"
#ifdef Q_OS_MACOS #include <QApplication>
#include "fileopeneventhandler.h" #include <QCommandLineParser>
#endif // Q_OS_MACOS #include <QDir>
#include <QTranslator>
#include <QApplication> #include <QUrl>
#include <QCommandLineParser>
#include <QDir> // QM_FILE_INSTALL_DIR should be defined from the CMakeLists file.
#include <QTranslator> #ifndef QM_FILE_INSTALL_DIR
#include <QUrl> #define QM_FILE_INSTALL_DIR ":/i18n/"
#endif // QM_FILE_INSTALL_DIR
using namespace Qt::Literals::StringLiterals;
int main(int argc, char *argv[])
int main(int argc, char *argv[]) {
{ QGuiApplication::setHighDpiScaleFactorRoundingPolicy(Settings::instance()->hiDpiScaleFactorBehavior());
QCoreApplication::setApplicationName(u"Pineapple Pictures"_s);
QCoreApplication::setApplicationVersion(PPIC_VERSION_STRING); QApplication a(argc, argv);
QGuiApplication::setHighDpiScaleFactorRoundingPolicy(Settings::instance()->hiDpiScaleFactorBehavior()); #if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
a.setAttribute(Qt::ApplicationAttribute::AA_UseHighDpiPixmaps);
QApplication a(argc, argv); #endif
QTranslator translator; QTranslator translator;
#if defined(TRANSLATION_RESOURCE_EMBEDDING) QString qmDir;
const QString qmDir = u":/i18n/"_s; #ifdef _WIN32
#elif defined(QM_FILE_INSTALL_ABSOLUTE_DIR) qmDir = QDir(QCoreApplication::applicationDirPath()).absoluteFilePath("translations");
const QString qmDir = QT_STRINGIFY(QM_FILE_INSTALL_ABSOLUTE_DIR); #else
#else qmDir = QT_STRINGIFY(QM_FILE_INSTALL_DIR);
const QString qmDir = QDir(QCoreApplication::applicationDirPath()).absoluteFilePath("translations"); #endif
#endif if (translator.load(QLocale(), QLatin1String("PineapplePictures"), QLatin1String("_"), qmDir)) {
if (translator.load(QLocale(), u"PineapplePictures"_s, u"_"_s, qmDir)) { a.installTranslator(&translator);
QCoreApplication::installTranslator(&translator); }
} a.setApplicationName("Pineapple Pictures");
a.setApplicationDisplayName(QCoreApplication::translate("main", "Pineapple Pictures"));
QGuiApplication::setApplicationDisplayName(QCoreApplication::translate("main", "Pineapple Pictures"));
// parse commandline arguments
// commandline options QCommandLineParser parser;
QCommandLineOption supportedImageFormats(u"supported-image-formats"_s, QCoreApplication::translate("main", "List supported image format suffixes, and quit program.")); parser.addPositionalArgument("File list", QCoreApplication::translate("main", "File list."));
// parse commandline arguments parser.addHelpOption();
QCommandLineParser parser;
parser.addOption(supportedImageFormats); parser.process(a);
parser.addPositionalArgument("File list", QCoreApplication::translate("main", "File list."));
parser.addHelpOption(); MainWindow w;
parser.process(a); w.show();
if (parser.isSet(supportedImageFormats)) { QStringList urlStrList = parser.positionalArguments();
#if QT_VERSION < QT_VERSION_CHECK(6, 9, 0) QList<QUrl> && urlList = PlaylistManager::convertToUrlList(urlStrList);
fputs(qPrintable(MainWindow::supportedImageFormats().join(QChar('\n'))), stdout);
::exit(EXIT_SUCCESS); if (!urlList.isEmpty()) {
#else w.showUrls(urlList);
QCommandLineParser::showMessageAndExit(QCommandLineParser::MessageType::Information, }
MainWindow::supportedImageFormats().join(QChar('\n')));
#endif w.initWindowSize();
}
return a.exec();
MainWindow w; }
w.show();
#ifdef Q_OS_MACOS
FileOpenEventHandler * fileOpenEventHandler = new FileOpenEventHandler(&a);
a.installEventFilter(fileOpenEventHandler);
a.connect(fileOpenEventHandler, &FileOpenEventHandler::fileOpen, [&w](const QUrl & url){
if (w.isHidden()) {
w.setWindowOpacity(1);
w.showNormal();
} else {
w.activateWindow();
}
w.showUrls({url});
w.initWindowSize();
});
// Handle dock icon clicks to show hidden window
a.connect(&a, &QApplication::applicationStateChanged, [&w](Qt::ApplicationState state) {
if (state == Qt::ApplicationActive && w.isHidden()) {
w.showUrls({});
w.galleryCurrent(true, true);
w.setWindowOpacity(1);
w.showNormal();
w.raise();
w.activateWindow();
}
});
#endif // Q_OS_MACOS
QStringList urlStrList = parser.positionalArguments();
QList<QUrl> && urlList = PlaylistManager::convertToUrlList(urlStrList);
if (!urlList.isEmpty()) {
w.showUrls(urlList);
}
w.initWindowSize();
return QApplication::exec();
}

View File

@ -1,4 +1,4 @@
// SPDX-FileCopyrightText: 2025 Gary Wang <git@blumia.net> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
@ -32,25 +32,19 @@
#include <QFile> #include <QFile>
#include <QTimer> #include <QTimer>
#include <QFileDialog> #include <QFileDialog>
#include <QFileSystemWatcher>
#include <QStandardPaths> #include <QStandardPaths>
#include <QStringBuilder>
#include <QProcess> #include <QProcess>
#include <QDesktopServices> #include <QDesktopServices>
#include <QMessageBox>
#ifdef HAVE_QTDBUS #ifdef HAVE_QTDBUS
#include <QDBusInterface> #include <QDBusInterface>
#include <QDBusConnectionInterface> #include <QDBusConnectionInterface>
#endif // HAVE_QTDBUS #endif // HAVE_QTDBUS
using namespace Qt::Literals::StringLiterals;
MainWindow::MainWindow(QWidget *parent) MainWindow::MainWindow(QWidget *parent)
: FramelessWindow(parent) : FramelessWindow(parent)
, m_am(new ActionManager) , m_am(new ActionManager)
, m_pm(new PlaylistManager(this)) , m_pm(new PlaylistManager(PlaylistManager::PL_SAMEFOLDER, this))
, m_fileSystemWatcher(new QFileSystemWatcher(this))
{ {
if (Settings::instance()->stayOnTop()) { if (Settings::instance()->stayOnTop()) {
this->setWindowFlag(Qt::WindowStaysOnTopHint); this->setWindowFlag(Qt::WindowStaysOnTopHint);
@ -58,24 +52,27 @@ MainWindow::MainWindow(QWidget *parent)
this->setAttribute(Qt::WA_TranslucentBackground, true); this->setAttribute(Qt::WA_TranslucentBackground, true);
this->setMinimumSize(350, 330); this->setMinimumSize(350, 330);
this->setWindowIcon(QIcon(u":/icons/app-icon.svg"_s)); this->setWindowIcon(QIcon(":/icons/app-icon.svg"));
this->setMouseTracking(true); this->setMouseTracking(true);
this->setAcceptDrops(true);
m_pm->setAutoLoadFilterSuffixes(supportedImageFormats()); m_pm->setAutoLoadFilterSuffix({
"*.jpg", "*.jpeg", "*.jfif",
"*.png", "*.gif", "*.svg", "*.bmp", "*.webp",
"*.tif", "*.tiff"
});
m_fadeOutAnimation = new QPropertyAnimation(this, "windowOpacity"_ba); m_fadeOutAnimation = new QPropertyAnimation(this, "windowOpacity");
m_fadeOutAnimation->setDuration(300); m_fadeOutAnimation->setDuration(300);
m_fadeOutAnimation->setStartValue(1); m_fadeOutAnimation->setStartValue(1);
m_fadeOutAnimation->setEndValue(0); m_fadeOutAnimation->setEndValue(0);
m_floatUpAnimation = new QPropertyAnimation(this, "geometry"_ba); m_floatUpAnimation = new QPropertyAnimation(this, "geometry");
m_floatUpAnimation->setDuration(300); m_floatUpAnimation->setDuration(300);
m_floatUpAnimation->setEasingCurve(QEasingCurve::OutCirc); m_floatUpAnimation->setEasingCurve(QEasingCurve::OutCirc);
m_exitAnimationGroup = new QParallelAnimationGroup(this); m_exitAnimationGroup = new QParallelAnimationGroup(this);
m_exitAnimationGroup->addAnimation(m_fadeOutAnimation); m_exitAnimationGroup->addAnimation(m_fadeOutAnimation);
m_exitAnimationGroup->addAnimation(m_floatUpAnimation); m_exitAnimationGroup->addAnimation(m_floatUpAnimation);
connect(m_exitAnimationGroup, &QParallelAnimationGroup::finished, connect(m_exitAnimationGroup, &QParallelAnimationGroup::finished,
this, &MainWindow::doCloseWindow); this, &QWidget::close);
GraphicsScene * scene = new GraphicsScene(this); GraphicsScene * scene = new GraphicsScene(this);
@ -90,7 +87,7 @@ MainWindow::MainWindow(QWidget *parent)
m_gv->fitInView(m_gv->sceneRect(), Qt::KeepAspectRatio); m_gv->fitInView(m_gv->sceneRect(), Qt::KeepAspectRatio);
connect(m_graphicsView, &GraphicsView::navigatorViewRequired, connect(m_graphicsView, &GraphicsView::navigatorViewRequired,
this, [this](bool required, const QTransform & tf){ this, [ = ](bool required, const QTransform & tf){
m_gv->setTransform(GraphicsView::resetScale(tf)); m_gv->setTransform(GraphicsView::resetScale(tf));
m_gv->fitInView(m_gv->sceneRect(), Qt::KeepAspectRatio); m_gv->fitInView(m_gv->sceneRect(), Qt::KeepAspectRatio);
m_gv->setVisible(required); m_gv->setVisible(required);
@ -100,6 +97,9 @@ MainWindow::MainWindow(QWidget *parent)
connect(m_graphicsView, &GraphicsView::viewportRectChanged, connect(m_graphicsView, &GraphicsView::viewportRectChanged,
m_gv, &NavigatorView::updateMainViewportRegion); m_gv, &NavigatorView::updateMainViewportRegion);
connect(m_graphicsView, &GraphicsView::requestGallery,
this, &MainWindow::loadGalleryBySingleLocalFile);
m_closeButton = new ToolButton(true, m_graphicsView); m_closeButton = new ToolButton(true, m_graphicsView);
m_closeButton->setIconSize(QSize(32, 32)); m_closeButton->setIconSize(QSize(32, 32));
m_closeButton->setFixedSize(QSize(50, 50)); m_closeButton->setFixedSize(QSize(50, 50));
@ -139,13 +139,18 @@ MainWindow::MainWindow(QWidget *parent)
m_gv->setOpacity(0, false); m_gv->setOpacity(0, false);
m_closeButton->setOpacity(0, false); m_closeButton->setOpacity(0, false);
connect(m_pm, &PlaylistManager::totalCountChanged, this, &MainWindow::updateGalleryButtonsVisibility); connect(m_pm, &PlaylistManager::loaded, this, [this](int galleryFileCount) {
m_prevButton->setVisible(galleryFileCount > 1);
m_nextButton->setVisible(galleryFileCount > 1);
});
connect(m_pm->model(), &PlaylistModel::modelReset, this, std::bind(&MainWindow::galleryCurrent, this, false, false)); connect(m_pm, &PlaylistManager::currentIndexChanged, this, [this]() {
connect(m_pm, &PlaylistManager::currentIndexChanged, this, std::bind(&MainWindow::galleryCurrent, this, true, false)); int index;
QUrl url;
connect(m_fileSystemWatcher, &QFileSystemWatcher::fileChanged, this, [this](){ std::tie(index, url) = m_pm->currentFileUrl();
QTimer::singleShot(500, this, std::bind(&MainWindow::galleryCurrent, this, false, true)); if (index != -1) {
this->setWindowTitle(url.fileName());
}
}); });
QShortcut * fullscreenShorucut = new QShortcut(QKeySequence(QKeySequence::FullScreen), this); QShortcut * fullscreenShorucut = new QShortcut(QKeySequence(QKeySequence::FullScreen), this);
@ -156,7 +161,6 @@ MainWindow::MainWindow(QWidget *parent)
QTimer::singleShot(0, this, [this](){ QTimer::singleShot(0, this, [this](){
m_am->setupShortcuts(); m_am->setupShortcuts();
Settings::instance()->applyUserShortcuts(this);
}); });
// allow some mouse events can go through these widgets for resizing window. // allow some mouse events can go through these widgets for resizing window.
@ -175,20 +179,15 @@ MainWindow::~MainWindow()
void MainWindow::showUrls(const QList<QUrl> &urls) void MainWindow::showUrls(const QList<QUrl> &urls)
{ {
if (!urls.isEmpty()) { if (!urls.isEmpty()) {
const QUrl & firstUrl = urls.first();
if (urls.count() == 1) { if (urls.count() == 1) {
const QString lowerCaseUrlPath(firstUrl.path().toLower()); m_graphicsView->showFileFromPath(urls.first().toLocalFile(), true);
if (lowerCaseUrlPath.endsWith(".m3u8") || lowerCaseUrlPath.endsWith(".m3u")) { } else {
m_pm->loadM3U8Playlist(firstUrl); m_graphicsView->showFileFromPath(urls.first().toLocalFile(), false);
galleryCurrent(true, true); m_pm->setPlaylist(urls);
return; m_pm->setCurrentIndex(0);
}
} }
m_graphicsView->showFileFromPath(firstUrl.toLocalFile());
m_pm->loadPlaylist(urls);
} else { } else {
m_graphicsView->showText(tr("File url list is empty")); m_graphicsView->showText(tr("File url list is empty"));
m_pm->setPlaylist(urls);
return; return;
} }
@ -204,9 +203,6 @@ void MainWindow::initWindowSize()
case Settings::WindowSizeBehavior::Maximized: case Settings::WindowSizeBehavior::Maximized:
showMaximized(); showMaximized();
break; break;
case Settings::WindowSizeBehavior::Windowed:
showNormal();
break;
default: default:
adjustWindowSizeBySceneRect(); adjustWindowSizeBySceneRect();
break; break;
@ -215,14 +211,14 @@ void MainWindow::initWindowSize()
void MainWindow::adjustWindowSizeBySceneRect() void MainWindow::adjustWindowSizeBySceneRect()
{ {
if (m_pm->totalCount() < 1) return; if (m_pm->count() < 1) return;
QSize sceneSize = m_graphicsView->sceneRect().toRect().size(); QSize sceneSize = m_graphicsView->sceneRect().toRect().size();
QSize sceneSizeWithMargins = sceneSize + QSize(130, 125); QSize sceneSizeWithMargins = sceneSize + QSize(130, 125);
if (m_graphicsView->scaleFactor() < 1 || size().expandedTo(sceneSizeWithMargins) != size()) { if (m_graphicsView->scaleFactor() < 1 || size().expandedTo(sceneSizeWithMargins) != size()) {
// if it scaled down by the resize policy: // if it scaled down by the resize policy:
QSize screenSize = window()->screen()->availableSize(); QSize screenSize = qApp->screenAt(QCursor::pos())->availableSize();
if (screenSize.expandedTo(sceneSize) == screenSize) { if (screenSize.expandedTo(sceneSize) == screenSize) {
// we can show the picture by increase the window size. // we can show the picture by increase the window size.
QSize finalSize = (screenSize.expandedTo(sceneSizeWithMargins) == screenSize) ? QSize finalSize = (screenSize.expandedTo(sceneSizeWithMargins) == screenSize) ?
@ -246,70 +242,46 @@ void MainWindow::adjustWindowSizeBySceneRect()
// can be empty if it is NOT from a local file. // can be empty if it is NOT from a local file.
QUrl MainWindow::currentImageFileUrl() const QUrl MainWindow::currentImageFileUrl() const
{ {
return m_pm->urlByIndex(m_pm->curIndex()); QUrl url;
std::tie(std::ignore, url) = m_pm->currentFileUrl();
return url;
} }
void MainWindow::clearGallery() void MainWindow::clearGallery()
{ {
m_pm->setPlaylist({}); m_pm->clear();
}
void MainWindow::loadGalleryBySingleLocalFile(const QString &path)
{
m_pm->setCurrentFile(path);
} }
void MainWindow::galleryPrev() void MainWindow::galleryPrev()
{ {
const bool loopGallery = Settings::instance()->loopGallery(); int index;
if (!loopGallery && m_pm->isFirstIndex()) return; QString filePath;
std::tie(index, filePath) = m_pm->previousFile();
QModelIndex index = m_pm->previousIndex(); if (index >= 0) {
if (index.isValid()) { m_graphicsView->showFileFromPath(filePath, false);
m_pm->setCurrentIndex(index); m_pm->setCurrentIndex(index);
m_graphicsView->showFileFromPath(m_pm->localFileByIndex(index));
} }
} }
void MainWindow::galleryNext() void MainWindow::galleryNext()
{ {
const bool loopGallery = Settings::instance()->loopGallery(); int index;
if (!loopGallery && m_pm->isLastIndex()) return; QString filePath;
std::tie(index, filePath) = m_pm->nextFile();
QModelIndex index = m_pm->nextIndex(); if (index >= 0) {
if (index.isValid()) { m_graphicsView->showFileFromPath(filePath, false);
m_pm->setCurrentIndex(index); m_pm->setCurrentIndex(index);
m_graphicsView->showFileFromPath(m_pm->localFileByIndex(index));
} }
} }
// Only use this to update minor information.
void MainWindow::galleryCurrent(bool showLoadImageHintWhenEmpty, bool reloadImage)
{
QModelIndex index = m_pm->curIndex();
bool shouldResetfileWatcher = true;
if (index.isValid()) {
const QString & localFilePath(m_pm->localFileByIndex(index));
if (reloadImage) m_graphicsView->showFileFromPath(localFilePath);
shouldResetfileWatcher = !updateFileWatcher(localFilePath);
setWindowTitle(m_pm->urlByIndex(index).fileName());
} else if (showLoadImageHintWhenEmpty && m_pm->totalCount() <= 0) {
m_graphicsView->showText(QCoreApplication::translate("GraphicsScene", "Drag image here"));
}
updateGalleryButtonsVisibility();
if (shouldResetfileWatcher) updateFileWatcher();
}
QStringList MainWindow::supportedImageFormats()
{
QStringList formatFilters {
#if QT_VERSION < QT_VERSION_CHECK(6, 8, 0)
QStringLiteral("*.jfif")
#endif // QT_VERSION < QT_VERSION_CHECK(6, 8, 0)
};
for (const QByteArray &item : QImageReader::supportedImageFormats()) {
formatFilters.append(QStringLiteral("*.") % QString::fromLocal8Bit(item));
}
return formatFilters;
}
void MainWindow::showEvent(QShowEvent *event) void MainWindow::showEvent(QShowEvent *event)
{ {
updateWidgetsPosition(); updateWidgetsPosition();
@ -317,7 +289,7 @@ void MainWindow::showEvent(QShowEvent *event)
return FramelessWindow::showEvent(event); return FramelessWindow::showEvent(event);
} }
void MainWindow::enterEvent(QEnterEvent *event) void MainWindow::enterEvent(QT_ENTER_EVENT *event)
{ {
m_bottomButtonGroup->setOpacity(1); m_bottomButtonGroup->setOpacity(1);
m_gv->setOpacity(1); m_gv->setOpacity(1);
@ -358,7 +330,11 @@ void MainWindow::mouseMoveEvent(QMouseEvent *event)
{ {
if (event->buttons() & Qt::LeftButton && m_clickedOnWindow && !isMaximized() && !isFullScreen()) { if (event->buttons() & Qt::LeftButton && m_clickedOnWindow && !isMaximized() && !isFullScreen()) {
if (!window()->windowHandle()->startSystemMove()) { if (!window()->windowHandle()->startSystemMove()) {
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
move(event->globalPosition().toPoint() - m_oldMousePos); move(event->globalPosition().toPoint() - m_oldMousePos);
#else
move(event->globalPos() - m_oldMousePos);
#endif // QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
} }
event->accept(); event->accept();
} }
@ -399,10 +375,6 @@ void MainWindow::mouseDoubleClickEvent(QMouseEvent *event)
toggleMaximize(); toggleMaximize();
event->accept(); event->accept();
break; break;
case Settings::DoubleClickBehavior::FullScreen:
toggleFullscreen();
event->accept();
break;
case Settings::DoubleClickBehavior::Ignore: case Settings::DoubleClickBehavior::Ignore:
break; break;
} }
@ -460,11 +432,12 @@ void MainWindow::contextMenuEvent(QContextMenuEvent *event)
QMenu * menu = new QMenu; QMenu * menu = new QMenu;
QMenu * copyMenu = new QMenu(tr("&Copy")); QMenu * copyMenu = new QMenu(tr("&Copy"));
QUrl currentFileUrl = currentImageFileUrl(); QUrl currentFileUrl = currentImageFileUrl();
QImage clipboardImage;
QUrl clipboardFileUrl;
QAction * copyPixmap = m_am->actionCopyPixmap; QAction * copyPixmap = m_am->actionCopyPixmap;
QAction * copyFilePath = m_am->actionCopyFilePath; QAction * copyFilePath = m_am->actionCopyFilePath;
copyMenu->setIcon(QIcon::fromTheme(u"edit-copy"_s));
copyMenu->addAction(copyPixmap); copyMenu->addAction(copyPixmap);
if (currentFileUrl.isValid()) { if (currentFileUrl.isValid()) {
copyMenu->addAction(copyFilePath); copyMenu->addAction(copyFilePath);
@ -472,8 +445,6 @@ void MainWindow::contextMenuEvent(QContextMenuEvent *event)
QAction * paste = m_am->actionPaste; QAction * paste = m_am->actionPaste;
QAction * trash = m_am->actionTrash;
QAction * stayOnTopMode = m_am->actionToggleStayOnTop; QAction * stayOnTopMode = m_am->actionToggleStayOnTop;
stayOnTopMode->setCheckable(true); stayOnTopMode->setCheckable(true);
stayOnTopMode->setChecked(stayOnTop()); stayOnTopMode->setChecked(stayOnTop());
@ -514,14 +485,15 @@ void MainWindow::contextMenuEvent(QContextMenuEvent *event)
menu->addSeparator(); menu->addSeparator();
menu->addAction(stayOnTopMode); menu->addAction(stayOnTopMode);
menu->addAction(protectedMode); menu->addAction(protectedMode);
#if 0
menu->addAction(avoidResetTransform); menu->addAction(avoidResetTransform);
#endif // 0
menu->addSeparator(); menu->addSeparator();
menu->addAction(toggleSettings); menu->addAction(toggleSettings);
menu->addAction(helpAction); menu->addAction(helpAction);
if (currentFileUrl.isValid()) { if (currentFileUrl.isValid()) {
menu->addSeparator(); menu->addSeparator();
if (currentFileUrl.isLocalFile()) { if (currentFileUrl.isLocalFile()) {
menu->addAction(trash);
menu->addAction(m_am->actionLocateInFileManager); menu->addAction(m_am->actionLocateInFileManager);
} }
menu->addAction(propertiesAction); menu->addAction(propertiesAction);
@ -533,50 +505,6 @@ void MainWindow::contextMenuEvent(QContextMenuEvent *event)
return FramelessWindow::contextMenuEvent(event); return FramelessWindow::contextMenuEvent(event);
} }
void MainWindow::dragEnterEvent(QDragEnterEvent *event)
{
if (event->mimeData()->hasUrls() || event->mimeData()->hasImage() || event->mimeData()->hasText()) {
event->acceptProposedAction();
} else {
event->ignore();
}
return FramelessWindow::dragEnterEvent(event);
}
void MainWindow::dragMoveEvent(QDragMoveEvent *event)
{
Q_UNUSED(event)
}
void MainWindow::dropEvent(QDropEvent *event)
{
event->acceptProposedAction();
const QMimeData * mimeData = event->mimeData();
if (mimeData->hasUrls()) {
const QList<QUrl> &urls = mimeData->urls();
if (urls.isEmpty()) {
m_graphicsView->showText(tr("File url list is empty"));
} else {
showUrls(urls);
}
} else if (mimeData->hasImage()) {
QImage img = qvariant_cast<QImage>(mimeData->imageData());
QPixmap pixmap = QPixmap::fromImage(img);
if (pixmap.isNull()) {
m_graphicsView->showText(tr("Image data is invalid"));
} else {
m_graphicsView->showImage(pixmap);
}
} else if (mimeData->hasText()) {
m_graphicsView->showText(mimeData->text());
} else {
m_graphicsView->showText(tr("Not supported mimedata: %1").arg(mimeData->formats().first()));
}
}
void MainWindow::centerWindow() void MainWindow::centerWindow()
{ {
this->setGeometry( this->setGeometry(
@ -584,23 +512,19 @@ void MainWindow::centerWindow()
Qt::LeftToRight, Qt::LeftToRight,
Qt::AlignCenter, Qt::AlignCenter,
this->size(), this->size(),
window()->screen()->availableGeometry() qApp->screenAt(QCursor::pos())->geometry()
) )
); );
} }
void MainWindow::closeWindow() void MainWindow::closeWindow()
{ {
if (Settings::instance()->useBuiltInCloseAnimation()) { QRect windowRect(this->geometry());
QRect windowRect(this->geometry()); m_floatUpAnimation->setStartValue(windowRect);
m_floatUpAnimation->setStartValue(windowRect); m_floatUpAnimation->setEndValue(windowRect.adjusted(0, -80, 0, 0));
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->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_floatUpAnimation->setEndValue(QRect(this->geometry().x(), this->geometry().y()-80, this->geometry().width(), this->geometry().height())); m_exitAnimationGroup->start();
m_exitAnimationGroup->start();
} else {
doCloseWindow();
}
} }
void MainWindow::updateWidgetsPosition() void MainWindow::updateWidgetsPosition()
@ -618,7 +542,8 @@ void MainWindow::toggleProtectedMode()
{ {
m_protectedMode = !m_protectedMode; m_protectedMode = !m_protectedMode;
m_closeButton->setVisible(!m_protectedMode); m_closeButton->setVisible(!m_protectedMode);
updateGalleryButtonsVisibility(); m_prevButton->setVisible(!m_protectedMode);
m_nextButton->setVisible(!m_protectedMode);
} }
void MainWindow::toggleStayOnTop() void MainWindow::toggleStayOnTop()
@ -772,34 +697,12 @@ void MainWindow::on_actionPaste_triggered()
} }
if (!clipboardImage.isNull()) { if (!clipboardImage.isNull()) {
setWindowTitle(tr("Image From Clipboard"));
m_graphicsView->showImage(clipboardImage); m_graphicsView->showImage(clipboardImage);
clearGallery(); clearGallery();
} else if (clipboardFileUrl.isValid()) { } else if (clipboardFileUrl.isValid()) {
m_graphicsView->showFileFromPath(clipboardFileUrl.toLocalFile()); QString localFile(clipboardFileUrl.toLocalFile());
m_pm->loadPlaylist(clipboardFileUrl); m_graphicsView->showFileFromPath(localFile, true);
} m_pm->setCurrentFile(localFile);
}
void MainWindow::on_actionTrash_triggered()
{
QModelIndex index = m_pm->curIndex();
if (!m_pm->urlByIndex(index).isLocalFile()) return;
QFile file(m_pm->localFileByIndex(index));
QFileInfo fileInfo(file.fileName());
QMessageBox::StandardButton result = QMessageBox::question(this, tr("Move to Trash"),
tr("Are you sure you want to move \"%1\" to recycle bin?").arg(fileInfo.fileName()));
if (result == QMessageBox::Yes) {
bool succ = file.moveToTrash();
if (!succ) {
QMessageBox::warning(this, tr("Failed to move file to trash"),
tr("Move to trash failed, it might caused by file permission issue, file system limitation, or platform limitation."));
} else {
m_pm->removeAt(index);
galleryCurrent(true, true);
}
} }
} }
@ -816,13 +719,6 @@ void MainWindow::on_actionRotateClockwise_triggered()
m_gv->setVisible(false); m_gv->setVisible(false);
} }
void MainWindow::on_actionRotateCounterClockwise_triggered()
{
m_graphicsView->rotateView(false);
m_graphicsView->displayScene();
m_gv->setVisible(false);
}
void MainWindow::on_actionPrevPicture_triggered() void MainWindow::on_actionPrevPicture_triggered()
{ {
galleryPrev(); galleryPrev();
@ -833,16 +729,6 @@ void MainWindow::on_actionNextPicture_triggered()
galleryNext(); galleryNext();
} }
void MainWindow::on_actionTogglePauseAnimation_triggered()
{
m_graphicsView->scene()->togglePauseAnimation();
}
void MainWindow::on_actionAnimationNextFrame_triggered()
{
m_graphicsView->scene()->skipAnimationFrame(1);
}
void MainWindow::on_actionToggleStayOnTop_triggered() void MainWindow::on_actionToggleStayOnTop_triggered()
{ {
toggleStayOnTop(); toggleStayOnTop();
@ -905,9 +791,9 @@ void MainWindow::on_actionLocateInFileManager_triggered()
QDesktopServices::openUrl(folderUrl); QDesktopServices::openUrl(folderUrl);
return; return;
} }
QDBusInterface fm1Iface(u"org.freedesktop.FileManager1"_s, QDBusInterface fm1Iface(QStringLiteral("org.freedesktop.FileManager1"),
u"/org/freedesktop/FileManager1"_s, QStringLiteral("/org/freedesktop/FileManager1"),
u"org.freedesktop.FileManager1"_s); QStringLiteral("org.freedesktop.FileManager1"));
fm1Iface.setTimeout(1000); fm1Iface.setTimeout(1000);
fm1Iface.callWithArgumentList(QDBus::Block, "ShowItems", { fm1Iface.callWithArgumentList(QDBus::Block, "ShowItems", {
QStringList{currentFileUrl.toString()}, QStringList{currentFileUrl.toString()},
@ -925,29 +811,3 @@ void MainWindow::on_actionQuitApp_triggered()
{ {
quitAppAction(false); quitAppAction(false);
} }
void MainWindow::doCloseWindow()
{
#ifdef Q_OS_MAC
this->hide();
#else
this->close();
#endif
}
bool MainWindow::updateFileWatcher(const QString &basePath)
{
m_fileSystemWatcher->removePaths(m_fileSystemWatcher->files());
if (!basePath.isEmpty()) return m_fileSystemWatcher->addPath(basePath);
return false;
}
void MainWindow::updateGalleryButtonsVisibility()
{
const int galleryFileCount = m_pm->totalCount();
const bool loopGallery = Settings::instance()->loopGallery();
m_prevButton->setVisible(!m_protectedMode && galleryFileCount > 1);
m_nextButton->setVisible(!m_protectedMode && galleryFileCount > 1);
m_prevButton->setEnabled(loopGallery || !m_pm->isFirstIndex());
m_nextButton->setEnabled(loopGallery || !m_pm->isLastIndex());
}

View File

@ -1,134 +1,122 @@
// SPDX-FileCopyrightText: 2025 Gary Wang <git@blumia.net> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#ifndef MAINWINDOW_H #ifndef MAINWINDOW_H
#define MAINWINDOW_H #define MAINWINDOW_H
#include "framelesswindow.h" #include "framelesswindow.h"
#include <QParallelAnimationGroup> #include <QParallelAnimationGroup>
#include <QPropertyAnimation> #include <QPropertyAnimation>
#include <QPushButton> #include <QPushButton>
QT_BEGIN_NAMESPACE #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
class QGraphicsOpacityEffect; typedef QEnterEvent QT_ENTER_EVENT;
class QGraphicsView; #else
class QFileSystemWatcher; typedef QEvent QT_ENTER_EVENT;
QT_END_NAMESPACE #endif // QT_VERSION_CHECK(6, 0, 0)
class ActionManager; QT_BEGIN_NAMESPACE
class PlaylistManager; class QGraphicsOpacityEffect;
class ToolButton; class QGraphicsView;
class GraphicsView; QT_END_NAMESPACE
class NavigatorView;
class BottomButtonGroup; class ActionManager;
class MainWindow : public FramelessWindow class PlaylistManager;
{ class ToolButton;
Q_OBJECT class GraphicsView;
class NavigatorView;
public: class BottomButtonGroup;
explicit MainWindow(QWidget *parent = nullptr); class MainWindow : public FramelessWindow
~MainWindow() override; {
Q_OBJECT
void showUrls(const QList<QUrl> &urls);
void initWindowSize(); public:
void adjustWindowSizeBySceneRect(); explicit MainWindow(QWidget *parent = nullptr);
QUrl currentImageFileUrl() const; ~MainWindow() override;
void clearGallery(); void showUrls(const QList<QUrl> &urls);
void galleryPrev(); void initWindowSize();
void galleryNext(); void adjustWindowSizeBySceneRect();
void galleryCurrent(bool showLoadImageHintWhenEmpty, bool reloadImage); QUrl currentImageFileUrl() const;
static QStringList supportedImageFormats(); void clearGallery();
void loadGalleryBySingleLocalFile(const QString &path);
protected slots: void galleryPrev();
void showEvent(QShowEvent *event) override; void galleryNext();
void enterEvent(QEnterEvent *event) override;
void leaveEvent(QEvent *event) override; protected slots:
void mousePressEvent(QMouseEvent *event) override; void showEvent(QShowEvent *event) override;
void mouseMoveEvent(QMouseEvent *event) override; void enterEvent(QT_ENTER_EVENT *event) override;
void mouseReleaseEvent(QMouseEvent *event) override; void leaveEvent(QEvent *event) override;
void mouseDoubleClickEvent(QMouseEvent *event) override; void mousePressEvent(QMouseEvent *event) override;
void wheelEvent(QWheelEvent *event) override; void mouseMoveEvent(QMouseEvent *event) override;
void resizeEvent(QResizeEvent *event) override; void mouseReleaseEvent(QMouseEvent *event) override;
void contextMenuEvent(QContextMenuEvent *event) override; void mouseDoubleClickEvent(QMouseEvent *event) override;
void dragEnterEvent(QDragEnterEvent *event) override; void wheelEvent(QWheelEvent *event) override;
void dragMoveEvent(QDragMoveEvent *event) override; void resizeEvent(QResizeEvent *event) override;
void dropEvent(QDropEvent *event) override; void contextMenuEvent(QContextMenuEvent *event) override;
void centerWindow(); void centerWindow();
void closeWindow(); void closeWindow();
void updateWidgetsPosition(); void updateWidgetsPosition();
void toggleProtectedMode(); void toggleProtectedMode();
void toggleStayOnTop(); void toggleStayOnTop();
void toggleAvoidResetTransform(); void toggleAvoidResetTransform();
bool stayOnTop() const; bool stayOnTop() const;
bool canPaste() const; bool canPaste() const;
void quitAppAction(bool force = false); void quitAppAction(bool force = false);
void toggleFullscreen(); void toggleFullscreen();
void toggleMaximize(); void toggleMaximize();
protected: protected:
QSize sizeHint() const override; QSize sizeHint() const override;
private slots: private slots:
void on_actionOpen_triggered(); void on_actionOpen_triggered();
void on_actionActualSize_triggered(); void on_actionActualSize_triggered();
void on_actionToggleMaximize_triggered(); void on_actionToggleMaximize_triggered();
void on_actionZoomIn_triggered(); void on_actionZoomIn_triggered();
void on_actionZoomOut_triggered(); void on_actionZoomOut_triggered();
void on_actionToggleCheckerboard_triggered(); void on_actionToggleCheckerboard_triggered();
void on_actionRotateClockwise_triggered(); void on_actionRotateClockwise_triggered();
void on_actionRotateCounterClockwise_triggered();
void on_actionPrevPicture_triggered();
void on_actionPrevPicture_triggered(); void on_actionNextPicture_triggered();
void on_actionNextPicture_triggered();
void on_actionHorizontalFlip_triggered();
void on_actionTogglePauseAnimation_triggered(); void on_actionFitInView_triggered();
void on_actionAnimationNextFrame_triggered(); void on_actionFitByWidth_triggered();
void on_actionCopyPixmap_triggered();
void on_actionHorizontalFlip_triggered(); void on_actionCopyFilePath_triggered();
void on_actionFitInView_triggered(); void on_actionPaste_triggered();
void on_actionFitByWidth_triggered(); void on_actionToggleStayOnTop_triggered();
void on_actionCopyPixmap_triggered(); void on_actionToggleProtectMode_triggered();
void on_actionCopyFilePath_triggered(); void on_actionToggleAvoidResetTransform_triggered();
void on_actionPaste_triggered(); void on_actionSettings_triggered();
void on_actionTrash_triggered(); void on_actionHelp_triggered();
void on_actionToggleStayOnTop_triggered(); void on_actionProperties_triggered();
void on_actionToggleProtectMode_triggered(); void on_actionLocateInFileManager_triggered();
void on_actionToggleAvoidResetTransform_triggered(); void on_actionQuitApp_triggered();
void on_actionSettings_triggered();
void on_actionHelp_triggered(); private:
void on_actionProperties_triggered(); ActionManager *m_am;
void on_actionLocateInFileManager_triggered(); PlaylistManager *m_pm;
void on_actionQuitApp_triggered();
QPoint m_oldMousePos;
void doCloseWindow(); QPropertyAnimation *m_fadeOutAnimation;
QPropertyAnimation *m_floatUpAnimation;
private: QParallelAnimationGroup *m_exitAnimationGroup;
bool updateFileWatcher(const QString & basePath = QString()); ToolButton *m_closeButton;
void updateGalleryButtonsVisibility(); ToolButton *m_prevButton;
ToolButton *m_nextButton;
private: GraphicsView *m_graphicsView;
ActionManager *m_am; NavigatorView *m_gv;
PlaylistManager *m_pm; BottomButtonGroup *m_bottomButtonGroup;
bool m_protectedMode = false;
QPoint m_oldMousePos; bool m_clickedOnWindow = false;
QPropertyAnimation *m_fadeOutAnimation; };
QPropertyAnimation *m_floatUpAnimation;
QParallelAnimationGroup *m_exitAnimationGroup; #endif // MAINWINDOW_H
QFileSystemWatcher *m_fileSystemWatcher;
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

View File

@ -1,110 +1,110 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "metadatadialog.h" #include "metadatadialog.h"
#include <QDialogButtonBox> #include <QDialogButtonBox>
#include <QPainter> #include <QPainter>
#include <QStyledItemDelegate> #include <QStyledItemDelegate>
#include <QTreeView> #include <QTreeView>
#include <QVBoxLayout> #include <QVBoxLayout>
#include <QHeaderView> #include <QHeaderView>
#include "metadatamodel.h" #include "metadatamodel.h"
class PropertyTreeView : public QTreeView class PropertyTreeView : public QTreeView
{ {
public: public:
explicit PropertyTreeView(QWidget* parent) : QTreeView(parent) {} explicit PropertyTreeView(QWidget* parent) : QTreeView(parent) {}
~PropertyTreeView() {} ~PropertyTreeView() {}
protected: protected:
void rowsInserted(const QModelIndex& parent, int start, int end) override void rowsInserted(const QModelIndex& parent, int start, int end) override
{ {
QTreeView::rowsInserted(parent, start, end); QTreeView::rowsInserted(parent, start, end);
if (!parent.isValid()) { if (!parent.isValid()) {
// we are inserting a section group // we are inserting a section group
for (int row = start; row <= end; ++row) { for (int row = start; row <= end; ++row) {
setupSection(row); setupSection(row);
} }
} else { } else {
// we are inserting a property // we are inserting a property
setRowHidden(parent.row(), QModelIndex(), false); setRowHidden(parent.row(), QModelIndex(), false);
} }
} }
void reset() override void reset() override
{ {
QTreeView::reset(); QTreeView::reset();
if (model()) { if (model()) {
for (int row = 0; row < model()->rowCount(); ++row) { for (int row = 0; row < model()->rowCount(); ++row) {
setupSection(row); setupSection(row);
} }
} }
} }
private: private:
void setupSection(int row) void setupSection(int row)
{ {
expand(model()->index(row, 0)); expand(model()->index(row, 0));
setFirstColumnSpanned(row, QModelIndex(), true); setFirstColumnSpanned(row, QModelIndex(), true);
setRowHidden(row, QModelIndex(), !model()->hasChildren(model()->index(row, 0))); setRowHidden(row, QModelIndex(), !model()->hasChildren(model()->index(row, 0)));
} }
}; };
class PropertyTreeItemDelegate : public QStyledItemDelegate class PropertyTreeItemDelegate : public QStyledItemDelegate
{ {
public: public:
PropertyTreeItemDelegate(QObject* parent) PropertyTreeItemDelegate(QObject* parent)
: QStyledItemDelegate(parent) : QStyledItemDelegate(parent)
{} {}
protected: protected:
void paint(QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index) const override void paint(QPainter* painter, const QStyleOptionViewItem& option, const QModelIndex& index) const override
{ {
QStyleOptionViewItem opt = option; QStyleOptionViewItem opt = option;
if (!index.parent().isValid()) { if (!index.parent().isValid()) {
opt.font.setBold(true); opt.font.setBold(true);
opt.features.setFlag(QStyleOptionViewItem::Alternate); opt.features.setFlag(QStyleOptionViewItem::Alternate);
} }
QStyledItemDelegate::paint(painter, opt, index); QStyledItemDelegate::paint(painter, opt, index);
} }
}; };
MetadataDialog::MetadataDialog(QWidget *parent) MetadataDialog::MetadataDialog(QWidget *parent)
: QDialog(parent) : QDialog(parent)
, m_treeView(new PropertyTreeView(this)) , m_treeView(new PropertyTreeView(this))
{ {
m_treeView->setRootIsDecorated(false); m_treeView->setRootIsDecorated(false);
m_treeView->setIndentation(0); m_treeView->setIndentation(0);
m_treeView->setItemDelegate(new PropertyTreeItemDelegate(m_treeView)); m_treeView->setItemDelegate(new PropertyTreeItemDelegate(m_treeView));
m_treeView->header()->resizeSection(0, sizeHint().width() / 2); m_treeView->header()->resizeSection(0, sizeHint().width() / 2);
setWindowTitle(tr("Image Metadata")); setWindowTitle(tr("Image Metadata"));
QDialogButtonBox * buttonBox = new QDialogButtonBox(QDialogButtonBox::Close); QDialogButtonBox * buttonBox = new QDialogButtonBox(QDialogButtonBox::Close);
setLayout(new QVBoxLayout); setLayout(new QVBoxLayout);
layout()->addWidget(m_treeView); layout()->addWidget(m_treeView);
layout()->addWidget(buttonBox); layout()->addWidget(buttonBox);
connect(buttonBox, &QDialogButtonBox::rejected, this, &QDialog::close); connect(buttonBox, &QDialogButtonBox::rejected, this, &QDialog::close);
setWindowFlag(Qt::WindowContextHelpButtonHint, false); setWindowFlag(Qt::WindowContextHelpButtonHint, false);
} }
MetadataDialog::~MetadataDialog() MetadataDialog::~MetadataDialog()
{ {
} }
void MetadataDialog::setMetadataModel(MetadataModel * model) void MetadataDialog::setMetadataModel(MetadataModel * model)
{ {
m_treeView->setModel(model); m_treeView->setModel(model);
} }
QSize MetadataDialog::sizeHint() const QSize MetadataDialog::sizeHint() const
{ {
return QSize(520, 350); return QSize(520, 350);
} }

View File

@ -1,30 +1,30 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#ifndef METADATADIALOG_H #ifndef METADATADIALOG_H
#define METADATADIALOG_H #define METADATADIALOG_H
#include <QDialog> #include <QDialog>
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE
class QTreeView; class QTreeView;
QT_END_NAMESPACE QT_END_NAMESPACE
class MetadataModel; class MetadataModel;
class MetadataDialog : public QDialog class MetadataDialog : public QDialog
{ {
Q_OBJECT Q_OBJECT
public: public:
explicit MetadataDialog(QWidget * parent); explicit MetadataDialog(QWidget * parent);
~MetadataDialog() override; ~MetadataDialog() override;
void setMetadataModel(MetadataModel * model); void setMetadataModel(MetadataModel * model);
QSize sizeHint() const override; QSize sizeHint() const override;
private: private:
QTreeView * m_treeView = nullptr; QTreeView * m_treeView = nullptr;
}; };
#endif // METADATADIALOG_H #endif // METADATADIALOG_H

View File

@ -1,320 +1,318 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "metadatamodel.h" #include "metadatamodel.h"
#include "exiv2wrapper.h" #include "exiv2wrapper.h"
#include <QDir> #include <QDir>
#include <QDebug> #include <QDebug>
#include <QDateTime> #include <QDateTime>
#include <QFileInfo> #include <QFileInfo>
#include <QImageReader> #include <QImageReader>
using namespace Qt::Literals::StringLiterals; MetadataModel::MetadataModel(QObject *parent)
: QAbstractItemModel(parent)
MetadataModel::MetadataModel(QObject *parent) {
: QAbstractItemModel(parent)
{ }
} MetadataModel::~MetadataModel()
{
MetadataModel::~MetadataModel()
{ }
} void MetadataModel::setFile(const QString &imageFilePath)
{
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.
QFileInfo fileInfo(imageFilePath); QImageReader imgReader(imageFilePath);
// It'll be fine if we don't re-use the image reader we pass to the graphics scene for now. imgReader.setAutoTransform(true);
QImageReader imgReader(imageFilePath); imgReader.setDecideFormatFromContent(true);
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 & itemTypeString = tr("%1 File").arg(QString(imgReader.format().toUpper())); const QString & birthTimeString = QLocale().toString(fileInfo.birthTime(), QLocale::LongFormat);
const QString & sizeString = QLocale().formattedDataSize(fileInfo.size()); const QString & lastModifiedTimeString = QLocale().toString(fileInfo.lastModified(), QLocale::LongFormat);
const QString & birthTimeString = QLocale().toString(fileInfo.birthTime(), QLocale::LongFormat); const QString & imageDimensionsString = imageSize(imgReader.size());
const QString & lastModifiedTimeString = QLocale().toString(fileInfo.lastModified(), QLocale::LongFormat); const QString & imageRatioString = imageSizeRatio(imgReader.size());
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(u"Description"_s, tr("Description", "Section name.")); appendSection(QStringLiteral("Image"), tr("Image", "Section name."));
appendSection(u"Origin"_s, tr("Origin", "Section name.")); appendSection(QStringLiteral("Camera"), tr("Camera", "Section name."));
appendSection(u"Image"_s, tr("Image", "Section name.")); appendSection(QStringLiteral("AdvancedPhoto"), tr("Advanced photo", "Section name."));
appendSection(u"Camera"_s, tr("Camera", "Section name.")); appendSection(QStringLiteral("GPS"), tr("GPS", "Section name."));
appendSection(u"AdvancedPhoto"_s, tr("Advanced photo", "Section name.")); appendSection(QStringLiteral("File"), tr("File", "Section name."));
appendSection(u"GPS"_s, tr("GPS", "Section name."));
appendSection(u"File"_s, tr("File", "Section name.")); if (imgReader.supportsOption(QImageIOHandler::Size)) {
appendProperty(QStringLiteral("Image"), QStringLiteral("Image.Dimensions"),
if (imgReader.supportsOption(QImageIOHandler::Size)) { tr("Dimensions"), imageDimensionsString);
appendProperty(u"Image"_s, u"Image.Dimensions"_s, appendProperty(QStringLiteral("Image"), QStringLiteral("Image.SizeRatio"),
tr("Dimensions"), imageDimensionsString); tr("Aspect ratio"), imageRatioString);
appendProperty(u"Image"_s, u"Image.SizeRatio"_s, }
tr("Aspect ratio"), imageRatioString); if (imgReader.supportsAnimation() && imgReader.imageCount() > 1) {
} appendProperty(QStringLiteral("Image"), QStringLiteral("Image.FrameCount"),
if (imgReader.supportsAnimation() && imgReader.imageCount() > 1) { tr("Frame count"), QString::number(imgReader.imageCount()));
appendProperty(u"Image"_s, u"Image.FrameCount"_s, }
tr("Frame count"), QString::number(imgReader.imageCount()));
} appendProperty(QStringLiteral("File"), QStringLiteral("File.Name"),
tr("Name"), fileInfo.fileName());
appendProperty(u"File"_s, u"File.Name"_s, appendProperty(QStringLiteral("File"), QStringLiteral("File.ItemType"),
tr("Name"), fileInfo.fileName()); tr("Item type"), itemTypeString);
appendProperty(u"File"_s, u"File.ItemType"_s, appendProperty(QStringLiteral("File"), QStringLiteral("File.Path"),
tr("Item type"), itemTypeString); tr("Folder path"), QDir::toNativeSeparators(fileInfo.path()));
appendProperty(u"File"_s, u"File.Path"_s, appendProperty(QStringLiteral("File"), QStringLiteral("File.Size"),
tr("Folder path"), QDir::toNativeSeparators(fileInfo.path())); tr("Size"), sizeString);
appendProperty(u"File"_s, u"File.Size"_s, appendProperty(QStringLiteral("File"), QStringLiteral("File.CreatedTime"),
tr("Size"), sizeString); tr("Date created"), birthTimeString);
appendProperty(u"File"_s, u"File.CreatedTime"_s, appendProperty(QStringLiteral("File"), QStringLiteral("File.LastModified"),
tr("Date created"), birthTimeString); tr("Date modified"), lastModifiedTimeString);
appendProperty(u"File"_s, u"File.LastModified"_s,
tr("Date modified"), lastModifiedTimeString); Exiv2Wrapper wrapper;
if (wrapper.load(imageFilePath)) {
Exiv2Wrapper wrapper; wrapper.cacheSections();
if (wrapper.load(imageFilePath)) {
wrapper.cacheSections(); appendExivPropertyIfExist(wrapper, QStringLiteral("Description"),
QStringLiteral("Xmp.dc.title"), tr("Title"), true);
appendExivPropertyIfExist(wrapper, u"Description"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Description"),
u"Xmp.dc.title"_s, tr("Title"), true); QStringLiteral("Exif.Image.ImageDescription"), tr("Subject"), true);
appendExivPropertyIfExist(wrapper, u"Description"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Description"),
u"Exif.Image.ImageDescription"_s, tr("Subject"), true); QStringLiteral("Exif.Image.Rating"), tr("Rating"));
appendExivPropertyIfExist(wrapper, u"Description"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Description"),
u"Exif.Image.Rating"_s, tr("Rating")); QStringLiteral("Xmp.dc.subject"), tr("Tags"));
appendExivPropertyIfExist(wrapper, u"Description"_s, appendPropertyIfNotEmpty(QStringLiteral("Description"), QStringLiteral("Description.Comments"),
u"Xmp.dc.subject"_s, tr("Tags")); tr("Comments"), wrapper.comment());
appendPropertyIfNotEmpty(u"Description"_s, u"Description.Comments"_s,
tr("Comments"), wrapper.comment()); appendExivPropertyIfExist(wrapper, QStringLiteral("Origin"),
QStringLiteral("Exif.Image.Artist"), tr("Authors"));
appendExivPropertyIfExist(wrapper, u"Origin"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Origin"),
u"Exif.Image.Artist"_s, tr("Authors")); QStringLiteral("Exif.Photo.DateTimeOriginal"), tr("Date taken"));
appendExivPropertyIfExist(wrapper, u"Origin"_s, // FIXME: We may fetch the same type of metadata from different metadata collection.
u"Exif.Photo.DateTimeOriginal"_s, tr("Date taken")); // Current implementation is not pretty and may need to do a rework...
// FIXME: We may fetch the same type of metadata from different metadata collection. // appendExivPropertyIfExist(wrapper, QStringLiteral("Origin"),
// Current implementation is not pretty and may need to do a rework... // QStringLiteral("Xmp.xmp.CreatorTool"), tr("Program name"));
// appendExivPropertyIfExist(wrapper, QStringLiteral("Origin"), appendExivPropertyIfExist(wrapper, QStringLiteral("Origin"),
// QStringLiteral("Xmp.xmp.CreatorTool"), tr("Program name")); QStringLiteral("Exif.Image.Software"), tr("Program name"));
appendExivPropertyIfExist(wrapper, u"Origin"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Origin"),
u"Exif.Image.Software"_s, tr("Program name")); QStringLiteral("Exif.Image.Copyright"), tr("Copyright"));
appendExivPropertyIfExist(wrapper, u"Origin"_s,
u"Exif.Image.Copyright"_s, tr("Copyright")); appendExivPropertyIfExist(wrapper, QStringLiteral("Image"),
QStringLiteral("Exif.Image.XResolution"), tr("Horizontal resolution"));
appendExivPropertyIfExist(wrapper, u"Image"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Image"),
u"Exif.Image.XResolution"_s, tr("Horizontal resolution")); QStringLiteral("Exif.Image.YResolution"), tr("Vertical resolution"));
appendExivPropertyIfExist(wrapper, u"Image"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Image"),
u"Exif.Image.YResolution"_s, tr("Vertical resolution")); QStringLiteral("Exif.Image.ResolutionUnit"), tr("Resolution unit"));
appendExivPropertyIfExist(wrapper, u"Image"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Image"),
u"Exif.Image.ResolutionUnit"_s, tr("Resolution unit")); QStringLiteral("Exif.Photo.ColorSpace"), tr("Colour representation"));
appendExivPropertyIfExist(wrapper, u"Image"_s,
u"Exif.Photo.ColorSpace"_s, tr("Colour representation")); appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"),
QStringLiteral("Exif.Image.Make"), tr("Camera maker"));
appendExivPropertyIfExist(wrapper, u"Camera"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"),
u"Exif.Image.Make"_s, tr("Camera maker")); QStringLiteral("Exif.Image.Model"), tr("Camera model"));
appendExivPropertyIfExist(wrapper, u"Camera"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"),
u"Exif.Image.Model"_s, tr("Camera model")); QStringLiteral("Exif.Photo.FNumber"), tr("F-stop"));
appendExivPropertyIfExist(wrapper, u"Camera"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"),
u"Exif.Photo.FNumber"_s, tr("F-stop")); QStringLiteral("Exif.Photo.ExposureTime"), tr("Exposure time"));
appendExivPropertyIfExist(wrapper, u"Camera"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"),
u"Exif.Photo.ExposureTime"_s, tr("Exposure time")); QStringLiteral("Exif.Photo.ISOSpeedRatings"), tr("ISO speed"));
appendExivPropertyIfExist(wrapper, u"Camera"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"),
u"Exif.Photo.ISOSpeedRatings"_s, tr("ISO speed")); QStringLiteral("Exif.Photo.ExposureBiasValue"), tr("Exposure bias"));
appendExivPropertyIfExist(wrapper, u"Camera"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"),
u"Exif.Photo.ExposureBiasValue"_s, tr("Exposure bias")); QStringLiteral("Exif.Photo.FocalLength"), tr("Focal length"));
appendExivPropertyIfExist(wrapper, u"Camera"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"),
u"Exif.Photo.FocalLength"_s, tr("Focal length")); QStringLiteral("Exif.Photo.MaxApertureValue"), tr("Max aperture"));
appendExivPropertyIfExist(wrapper, u"Camera"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"),
u"Exif.Photo.MaxApertureValue"_s, tr("Max aperture")); QStringLiteral("Exif.Photo.MeteringMode"), tr("Metering mode"));
appendExivPropertyIfExist(wrapper, u"Camera"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"),
u"Exif.Photo.MeteringMode"_s, tr("Metering mode")); QStringLiteral("Exif.Photo.SubjectDistance"), tr("Subject distance"));
appendExivPropertyIfExist(wrapper, u"Camera"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"),
u"Exif.Photo.SubjectDistance"_s, tr("Subject distance")); QStringLiteral("Exif.Photo.Flash"), tr("Flash mode"));
appendExivPropertyIfExist(wrapper, u"Camera"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("Camera"),
u"Exif.Photo.Flash"_s, tr("Flash mode")); QStringLiteral("Exif.Photo.FocalLengthIn35mmFilm"), tr("35mm focal length"));
appendExivPropertyIfExist(wrapper, u"Camera"_s,
u"Exif.Photo.FocalLengthIn35mmFilm"_s, tr("35mm focal length")); appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"),
QStringLiteral("Exif.Photo.LensModel"), tr("Lens model"));
appendExivPropertyIfExist(wrapper, u"AdvancedPhoto"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"),
u"Exif.Photo.LensModel"_s, tr("Lens model")); QStringLiteral("Exif.Photo.Contrast"), tr("Contrast"));
appendExivPropertyIfExist(wrapper, u"AdvancedPhoto"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"),
u"Exif.Photo.Contrast"_s, tr("Contrast")); QStringLiteral("Exif.Photo.BrightnessValue"), tr("Brightness"));
appendExivPropertyIfExist(wrapper, u"AdvancedPhoto"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"),
u"Exif.Photo.BrightnessValue"_s, tr("Brightness")); QStringLiteral("Exif.Photo.ExposureProgram"), tr("Exposure program"));
appendExivPropertyIfExist(wrapper, u"AdvancedPhoto"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"),
u"Exif.Photo.ExposureProgram"_s, tr("Exposure program")); QStringLiteral("Exif.Photo.Saturation"), tr("Saturation"));
appendExivPropertyIfExist(wrapper, u"AdvancedPhoto"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"),
u"Exif.Photo.Saturation"_s, tr("Saturation")); QStringLiteral("Exif.Photo.Sharpness"), tr("Sharpness"));
appendExivPropertyIfExist(wrapper, u"AdvancedPhoto"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"),
u"Exif.Photo.Sharpness"_s, tr("Sharpness")); QStringLiteral("Exif.Photo.WhiteBalance"), tr("White balance"));
appendExivPropertyIfExist(wrapper, u"AdvancedPhoto"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"),
u"Exif.Photo.WhiteBalance"_s, tr("White balance")); QStringLiteral("Exif.Photo.DigitalZoomRatio"), tr("Digital zoom"));
appendExivPropertyIfExist(wrapper, u"AdvancedPhoto"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("AdvancedPhoto"),
u"Exif.Photo.DigitalZoomRatio"_s, tr("Digital zoom")); QStringLiteral("Exif.Photo.ExifVersion"), tr("EXIF version"));
appendExivPropertyIfExist(wrapper, u"AdvancedPhoto"_s,
u"Exif.Photo.ExifVersion"_s, tr("EXIF version")); appendExivPropertyIfExist(wrapper, QStringLiteral("GPS"),
QStringLiteral("Exif.GPSInfo.GPSLatitudeRef"), tr("Latitude reference"));
appendExivPropertyIfExist(wrapper, u"GPS"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("GPS"),
u"Exif.GPSInfo.GPSLatitudeRef"_s, tr("Latitude reference")); QStringLiteral("Exif.GPSInfo.GPSLatitude"), tr("Latitude"));
appendExivPropertyIfExist(wrapper, u"GPS"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("GPS"),
u"Exif.GPSInfo.GPSLatitude"_s, tr("Latitude")); QStringLiteral("Exif.GPSInfo.GPSLongitudeRef"), tr("Longitude reference"));
appendExivPropertyIfExist(wrapper, u"GPS"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("GPS"),
u"Exif.GPSInfo.GPSLongitudeRef"_s, tr("Longitude reference")); QStringLiteral("Exif.GPSInfo.GPSLongitude"), tr("Longitude"));
appendExivPropertyIfExist(wrapper, u"GPS"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("GPS"),
u"Exif.GPSInfo.GPSLongitude"_s, tr("Longitude")); QStringLiteral("Exif.GPSInfo.GPSAltitudeRef"), tr("Altitude reference"));
appendExivPropertyIfExist(wrapper, u"GPS"_s, appendExivPropertyIfExist(wrapper, QStringLiteral("GPS"),
u"Exif.GPSInfo.GPSAltitudeRef"_s, tr("Altitude reference")); QStringLiteral("Exif.GPSInfo.GPSAltitude"), tr("Altitude"));
appendExivPropertyIfExist(wrapper, u"GPS"_s,
u"Exif.GPSInfo.GPSAltitude"_s, tr("Altitude")); }
}
}
} QString MetadataModel::imageSize(const QSize &size)
{
QString MetadataModel::imageSize(const QSize &size) QString imageSize;
{
QString imageSize; if (size.isValid()) {
imageSize = tr("%1 x %2").arg(QString::number(size.width()), QString::number(size.height()));
if (size.isValid()) { } else {
imageSize = tr("%1 x %2").arg(QString::number(size.width()), QString::number(size.height())); imageSize = QLatin1Char('-');
} else { }
imageSize = QLatin1Char('-');
} return imageSize;
}
return imageSize;
} int simplegcd(int a, int b) {
return b == 0 ? a : simplegcd(b, a % b);
int simplegcd(int a, int b) { }
return b == 0 ? a : simplegcd(b, a % b);
} QString MetadataModel::imageSizeRatio(const QSize &size)
{
QString MetadataModel::imageSizeRatio(const QSize &size) if (!size.isValid()) {
{ return QStringLiteral("-");
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));
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 &sectionKey, QStringView sectionDisplayName)
{
bool MetadataModel::appendSection(const QString &sectionKey, QStringView sectionDisplayName) if (m_sections.contains(sectionKey)) {
{ return false;
if (m_sections.contains(sectionKey)) { }
return false;
} m_sections.append(sectionKey);
m_sectionProperties[sectionKey] = qMakePair<QString, QList<QString> >(sectionDisplayName.toString(), {});
m_sections.append(sectionKey);
m_sectionProperties[sectionKey] = qMakePair<QString, QList<QString> >(sectionDisplayName.toString(), {}); return true;
}
return true;
} bool MetadataModel::appendPropertyIfNotEmpty(const QString &sectionKey, const QString &propertyKey, const QString &propertyDisplayName, const QString &propertyValue)
{
bool MetadataModel::appendPropertyIfNotEmpty(const QString &sectionKey, const QString &propertyKey, const QString &propertyDisplayName, const QString &propertyValue) if (propertyValue.isEmpty()) return false;
{
if (propertyValue.isEmpty()) return false; return appendProperty(sectionKey, propertyKey, propertyDisplayName, propertyValue);
}
return appendProperty(sectionKey, propertyKey, propertyDisplayName, propertyValue);
} bool MetadataModel::appendProperty(const QString &sectionKey, const QString &propertyKey, QStringView propertyDisplayName, QStringView propertyValue)
{
bool MetadataModel::appendProperty(const QString &sectionKey, const QString &propertyKey, QStringView propertyDisplayName, QStringView propertyValue) if (!m_sections.contains(sectionKey)) {
{ return false;
if (!m_sections.contains(sectionKey)) { }
return false;
} QList<QString> & propertyList = m_sectionProperties[sectionKey].second;
if (!propertyList.contains(propertyKey)) {
QList<QString> & propertyList = m_sectionProperties[sectionKey].second; propertyList.append(propertyKey);
if (!propertyList.contains(propertyKey)) { }
propertyList.append(propertyKey);
} m_properties[propertyKey] = qMakePair<QString, QString>(propertyDisplayName.toString(), propertyValue.toString());
m_properties[propertyKey] = qMakePair<QString, QString>(propertyDisplayName.toString(), propertyValue.toString()); return true;
}
return true;
} bool MetadataModel::appendExivPropertyIfExist(const Exiv2Wrapper &wrapper, const QString &sectionKey, const QString &exiv2propertyKey, const QString &propertyDisplayName, bool isXmpString)
{
bool MetadataModel::appendExivPropertyIfExist(const Exiv2Wrapper &wrapper, const QString &sectionKey, const QString &exiv2propertyKey, const QString &propertyDisplayName, bool isXmpString) const QString & value = wrapper.value(exiv2propertyKey);
{ if (!value.isEmpty()) {
const QString & value = wrapper.value(exiv2propertyKey); appendProperty(sectionKey, exiv2propertyKey,
if (!value.isEmpty()) { propertyDisplayName.isEmpty() ? wrapper.label(exiv2propertyKey) : propertyDisplayName,
appendProperty(sectionKey, exiv2propertyKey, isXmpString ? Exiv2Wrapper::XmpValue(value) : value);
propertyDisplayName.isEmpty() ? wrapper.label(exiv2propertyKey) : propertyDisplayName, return true;
isXmpString ? Exiv2Wrapper::XmpValue(value) : value); }
return true; return false;
} }
return false;
} QModelIndex MetadataModel::index(int row, int column, const QModelIndex &parent) const
{
QModelIndex MetadataModel::index(int row, int column, const QModelIndex &parent) const if (!hasIndex(row, column, parent)) {
{ return QModelIndex();
if (!hasIndex(row, column, parent)) { }
return QModelIndex();
} if (!parent.isValid()) {
return createIndex(row, column, RowType::SectionRow);
if (!parent.isValid()) { } else {
return createIndex(row, column, RowType::SectionRow); // internalid param: row means nth section it belongs to.
} else { return createIndex(row, column, RowType::PropertyRow + parent.row());
// internalid param: row means nth section it belongs to. }
return createIndex(row, column, RowType::PropertyRow + parent.row()); }
}
} QModelIndex MetadataModel::parent(const QModelIndex &child) const
{
QModelIndex MetadataModel::parent(const QModelIndex &child) const if (!child.isValid()) {
{ return QModelIndex();
if (!child.isValid()) { }
return QModelIndex();
} if (child.internalId() == RowType::SectionRow) {
return QModelIndex();
if (child.internalId() == RowType::SectionRow) { } else {
return QModelIndex(); return createIndex(child.internalId() - RowType::PropertyRow, 0, SectionRow);
} else { }
return createIndex(child.internalId() - RowType::PropertyRow, 0, SectionRow); }
}
} int MetadataModel::rowCount(const QModelIndex &parent) const
{
int MetadataModel::rowCount(const QModelIndex &parent) const if (!parent.isValid()) {
{ return m_sections.count();
if (!parent.isValid()) { }
return m_sections.count();
} if (parent.internalId() == RowType::SectionRow) {
const QString & sectionKey = m_sections[parent.row()];
if (parent.internalId() == RowType::SectionRow) { return m_sectionProperties[sectionKey].second.count();
const QString & sectionKey = m_sections[parent.row()]; }
return m_sectionProperties[sectionKey].second.count();
} return 0;
}
return 0;
} int MetadataModel::columnCount(const QModelIndex &) const
{
int MetadataModel::columnCount(const QModelIndex &) const // Always key(display name) and value.
{ return 2;
// Always key(display name) and value. }
return 2;
} QVariant MetadataModel::data(const QModelIndex &index, int role) const
{
QVariant MetadataModel::data(const QModelIndex &index, int role) const if (!index.isValid()) {
{ return QVariant();
if (!index.isValid()) { }
return QVariant();
} if (role != Qt::DisplayRole) {
return QVariant();
if (role != Qt::DisplayRole) { }
return QVariant();
} if (index.internalId() == RowType::SectionRow) {
return (index.column() == 0) ? m_sectionProperties[m_sections[index.row()]].first
if (index.internalId() == RowType::SectionRow) { : QVariant();
return (index.column() == 0) ? m_sectionProperties[m_sections[index.row()]].first } else {
: QVariant(); int sectionIndex = index.internalId() - RowType::PropertyRow;
} else { const QString & sectionKey = m_sections[sectionIndex];
int sectionIndex = index.internalId() - RowType::PropertyRow; const QList<QString> & propertyList = m_sectionProperties[sectionKey].second;
const QString & sectionKey = m_sections[sectionIndex]; return (index.column() == 0) ? m_properties[propertyList[index.row()]].first
const QList<QString> & propertyList = m_sectionProperties[sectionKey].second; : m_properties[propertyList[index.row()]].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
{
QVariant MetadataModel::headerData(int section, Qt::Orientation orientation, int role) const if (orientation == Qt::Vertical || role != Qt::DisplayRole) {
{ return QVariant();
if (orientation == Qt::Vertical || role != Qt::DisplayRole) { }
return QVariant();
} return section == 0 ? tr("Property") : tr("Value");
}
return section == 0 ? tr("Property") : tr("Value");
}

View File

@ -1,52 +1,52 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#ifndef METADATAMODEL_H #ifndef METADATAMODEL_H
#define METADATAMODEL_H #define METADATAMODEL_H
#include <QAbstractItemModel> #include <QAbstractItemModel>
class Exiv2Wrapper; class Exiv2Wrapper;
class MetadataModel : public QAbstractItemModel class MetadataModel : public QAbstractItemModel
{ {
Q_OBJECT Q_OBJECT
public: public:
explicit MetadataModel(QObject *parent = nullptr); explicit MetadataModel(QObject *parent = nullptr);
~MetadataModel(); ~MetadataModel();
void setFile(const QString & imageFilePath); void setFile(const QString & imageFilePath);
static QString imageSize(const QSize &size); static QString imageSize(const QSize &size);
static QString imageSizeRatio(const QSize &size); static QString imageSizeRatio(const QSize &size);
bool appendSection(const QString & sectionKey, QStringView sectionDisplayName); bool appendSection(const QString & sectionKey, QStringView sectionDisplayName);
bool appendPropertyIfNotEmpty(const QString & sectionKey, const QString & propertyKey, bool appendPropertyIfNotEmpty(const QString & sectionKey, const QString & propertyKey,
const QString & propertyDisplayName, const QString & propertyValue = QString()); const QString & propertyDisplayName, const QString & propertyValue = QString());
bool appendProperty(const QString & sectionKey, const QString & propertyKey, bool appendProperty(const QString & sectionKey, const QString & propertyKey,
QStringView propertyDisplayName, QStringView propertyValue = QString()); QStringView propertyDisplayName, QStringView propertyValue = QString());
bool appendExivPropertyIfExist(const Exiv2Wrapper & wrapper, const QString & sectionKey, bool appendExivPropertyIfExist(const Exiv2Wrapper & wrapper, const QString & sectionKey,
const QString & exiv2propertyKey, const QString & propertyDisplayName = QString(), const QString & exiv2propertyKey, const QString & propertyDisplayName = QString(),
bool isXmpString = false); bool isXmpString = false);
private: private:
enum RowType : quintptr { enum RowType : quintptr {
SectionRow, SectionRow,
PropertyRow, PropertyRow,
}; };
QModelIndex index(int row, int column, const QModelIndex &parent = QModelIndex()) const override; QModelIndex index(int row, int column, const QModelIndex &parent = QModelIndex()) const override;
QModelIndex parent(const QModelIndex &child) const override; QModelIndex parent(const QModelIndex &child) const override;
int rowCount(const QModelIndex &parent = QModelIndex()) const override; int rowCount(const QModelIndex &parent = QModelIndex()) const override;
int columnCount(const QModelIndex & = QModelIndex()) const override; int columnCount(const QModelIndex & = QModelIndex()) const override;
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) 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; QVariant headerData(int section, Qt::Orientation orientation, int role = Qt::DisplayRole) const override;
// [SECTION_KEY] // [SECTION_KEY]
QList<QString> m_sections; QList<QString> m_sections;
// {SECTION_KEY: (SECTION_DISPLAY_NAME, [PROPERTY_KEY])} // {SECTION_KEY: (SECTION_DISPLAY_NAME, [PROPERTY_KEY])}
QMap<QString, QPair<QString, QList<QString> > > m_sectionProperties; QMap<QString, QPair<QString, QList<QString> > > m_sectionProperties;
// {PROPERTY_KEY: (PROPERTY_DISPLAY_NAME, PROPERTY_VALUE)} // {PROPERTY_KEY: (PROPERTY_DISPLAY_NAME, PROPERTY_VALUE)}
QMap<QString, QPair<QString, QString> > m_properties; QMap<QString, QPair<QString, QString> > m_properties;
}; };
#endif // METADATAMODEL_H #endif // METADATAMODEL_H

View File

@ -1,86 +1,86 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "navigatorview.h" #include "navigatorview.h"
#include "graphicsview.h" #include "graphicsview.h"
#include "opacityhelper.h" #include "opacityhelper.h"
#include <QMouseEvent> #include <QMouseEvent>
#include <QDebug> #include <QDebug>
NavigatorView::NavigatorView(QWidget *parent) NavigatorView::NavigatorView(QWidget *parent)
: QGraphicsView (parent) : QGraphicsView (parent)
, m_viewportRegion(this->rect()) , m_viewportRegion(this->rect())
, m_opacityHelper(new OpacityHelper(this)) , m_opacityHelper(new OpacityHelper(this))
{ {
setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
setStyleSheet("background-color: rgba(0, 0, 0, 120);" setStyleSheet("background-color: rgba(0, 0, 0, 120);"
"border-radius: 3px;"); "border-radius: 3px;");
} }
// doesn't take or manage its ownership // doesn't take or manage its ownership
void NavigatorView::setMainView(GraphicsView *mainView) void NavigatorView::setMainView(GraphicsView *mainView)
{ {
m_mainView = mainView; m_mainView = mainView;
} }
void NavigatorView::setOpacity(qreal opacity, bool animated) void NavigatorView::setOpacity(qreal opacity, bool animated)
{ {
m_opacityHelper->setOpacity(opacity, animated); m_opacityHelper->setOpacity(opacity, animated);
} }
void NavigatorView::updateMainViewportRegion() void NavigatorView::updateMainViewportRegion()
{ {
if (m_mainView != nullptr) { if (m_mainView != nullptr) {
m_viewportRegion = mapFromScene(m_mainView->mapToScene(m_mainView->rect())); m_viewportRegion = mapFromScene(m_mainView->mapToScene(m_mainView->rect()));
update(); update();
} }
} }
void NavigatorView::mousePressEvent(QMouseEvent *event) void NavigatorView::mousePressEvent(QMouseEvent *event)
{ {
m_mouseDown = true; m_mouseDown = true;
if (m_mainView) { if (m_mainView) {
m_mainView->centerOn(mapToScene(event->pos())); m_mainView->centerOn(mapToScene(event->pos()));
update(); update();
} }
event->accept(); event->accept();
} }
void NavigatorView::mouseMoveEvent(QMouseEvent *event) void NavigatorView::mouseMoveEvent(QMouseEvent *event)
{ {
if (m_mouseDown && m_mainView) { if (m_mouseDown && m_mainView) {
m_mainView->centerOn(mapToScene(event->pos())); m_mainView->centerOn(mapToScene(event->pos()));
update(); update();
event->accept(); event->accept();
} else { } else {
event->ignore(); event->ignore();
} }
} }
void NavigatorView::mouseReleaseEvent(QMouseEvent *event) void NavigatorView::mouseReleaseEvent(QMouseEvent *event)
{ {
m_mouseDown = false; m_mouseDown = false;
event->accept(); event->accept();
} }
void NavigatorView::wheelEvent(QWheelEvent *event) void NavigatorView::wheelEvent(QWheelEvent *event)
{ {
event->ignore(); event->ignore();
return QGraphicsView::wheelEvent(event); return QGraphicsView::wheelEvent(event);
} }
void NavigatorView::paintEvent(QPaintEvent *event) void NavigatorView::paintEvent(QPaintEvent *event)
{ {
QGraphicsView::paintEvent(event); QGraphicsView::paintEvent(event);
QPainter painter(viewport()); QPainter painter(viewport());
painter.setPen(QPen(Qt::gray, 2)); painter.setPen(QPen(Qt::gray, 2));
painter.drawRect(m_viewportRegion.boundingRect()); painter.drawRect(m_viewportRegion.boundingRect());
} }

View File

@ -1,38 +1,38 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#ifndef NAVIGATORVIEW_H #ifndef NAVIGATORVIEW_H
#define NAVIGATORVIEW_H #define NAVIGATORVIEW_H
#include <QGraphicsView> #include <QGraphicsView>
class OpacityHelper; class OpacityHelper;
class GraphicsView; class GraphicsView;
class NavigatorView : public QGraphicsView class NavigatorView : public QGraphicsView
{ {
Q_OBJECT Q_OBJECT
public: public:
NavigatorView(QWidget *parent = nullptr); NavigatorView(QWidget *parent = nullptr);
void setMainView(GraphicsView *mainView); void setMainView(GraphicsView *mainView);
void setOpacity(qreal opacity, bool animated = true); void setOpacity(qreal opacity, bool animated = true);
public slots: public slots:
void updateMainViewportRegion(); void updateMainViewportRegion();
private: private:
void mousePressEvent(QMouseEvent * event) override; void mousePressEvent(QMouseEvent * event) override;
void mouseMoveEvent(QMouseEvent * event) override; void mouseMoveEvent(QMouseEvent * event) override;
void mouseReleaseEvent(QMouseEvent * event) override; void mouseReleaseEvent(QMouseEvent * event) override;
void wheelEvent(QWheelEvent *event) override; void wheelEvent(QWheelEvent *event) override;
void paintEvent(QPaintEvent *event) override; void paintEvent(QPaintEvent *event) override;
bool m_mouseDown = false; bool m_mouseDown = false;
QPolygon m_viewportRegion; QPolygon m_viewportRegion;
QGraphicsView *m_mainView = nullptr; QGraphicsView *m_mainView = nullptr;
OpacityHelper *m_opacityHelper = nullptr; OpacityHelper *m_opacityHelper = nullptr;
}; };
#endif // NAVIGATORVIEW_H #endif // NAVIGATORVIEW_H

View File

@ -1,31 +1,31 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "opacityhelper.h" #include "opacityhelper.h"
#include <QGraphicsOpacityEffect> #include <QGraphicsOpacityEffect>
#include <QPropertyAnimation> #include <QPropertyAnimation>
OpacityHelper::OpacityHelper(QWidget *parent) OpacityHelper::OpacityHelper(QWidget *parent)
: QObject(parent) : QObject(parent)
, m_opacityFx(new QGraphicsOpacityEffect(parent)) , m_opacityFx(new QGraphicsOpacityEffect(parent))
, m_opacityAnimation(new QPropertyAnimation(m_opacityFx, "opacity")) , m_opacityAnimation(new QPropertyAnimation(m_opacityFx, "opacity"))
{ {
parent->setGraphicsEffect(m_opacityFx); parent->setGraphicsEffect(m_opacityFx);
m_opacityAnimation->setDuration(300); m_opacityAnimation->setDuration(300);
} }
void OpacityHelper::setOpacity(qreal opacity, bool animated) void OpacityHelper::setOpacity(qreal opacity, bool animated)
{ {
if (!animated) { if (!animated) {
m_opacityFx->setOpacity(opacity); m_opacityFx->setOpacity(opacity);
return; return;
} }
m_opacityAnimation->stop(); m_opacityAnimation->stop();
m_opacityAnimation->setStartValue(m_opacityFx->opacity()); m_opacityAnimation->setStartValue(m_opacityFx->opacity());
m_opacityAnimation->setEndValue(opacity); m_opacityAnimation->setEndValue(opacity);
m_opacityAnimation->start(); m_opacityAnimation->start();
} }

View File

@ -1,27 +1,27 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#ifndef OPACITYHELPER_H #ifndef OPACITYHELPER_H
#define OPACITYHELPER_H #define OPACITYHELPER_H
#include <QWidget> #include <QWidget>
QT_BEGIN_NAMESPACE QT_BEGIN_NAMESPACE
class QGraphicsOpacityEffect; class QGraphicsOpacityEffect;
class QPropertyAnimation; class QPropertyAnimation;
QT_END_NAMESPACE QT_END_NAMESPACE
class OpacityHelper : QObject class OpacityHelper : QObject
{ {
public: public:
OpacityHelper(QWidget * parent); OpacityHelper(QWidget * parent);
void setOpacity(qreal opacity, bool animated = true); void setOpacity(qreal opacity, bool animated = true);
protected: protected:
QGraphicsOpacityEffect * m_opacityFx; QGraphicsOpacityEffect * m_opacityFx;
QPropertyAnimation * m_opacityAnimation; QPropertyAnimation * m_opacityAnimation;
}; };
#endif // OPACITYHELPER_H #endif // OPACITYHELPER_H

View File

@ -1,283 +1,175 @@
// SPDX-FileCopyrightText: 2025 Gary Wang <git@blumia.net> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "playlistmanager.h" #include "playlistmanager.h"
#include <QCollator> #include <QCollator>
#include <QDir> #include <QDir>
#include <QFileInfo> #include <QFileInfo>
#include <QUrl> #include <QUrl>
PlaylistModel::PlaylistModel(QObject *parent) PlaylistManager::PlaylistManager(PlaylistType type, QObject *parent)
: QAbstractListModel(parent) : QObject(parent)
{ , m_type(type)
{
}
}
PlaylistModel::~PlaylistModel()
= default; PlaylistManager::~PlaylistManager()
{
void PlaylistModel::setPlaylist(const QList<QUrl> &urls)
{ }
beginResetModel();
m_playlist = urls; void PlaylistManager::setPlaylistType(PlaylistManager::PlaylistType type)
endResetModel(); {
} m_type = type;
}
QModelIndex PlaylistModel::loadPlaylist(const QList<QUrl> & urls)
{ PlaylistManager::PlaylistType PlaylistManager::playlistType() const
if (urls.isEmpty()) return {}; {
if (urls.count() == 1) { return m_type;
return loadPlaylist(urls.constFirst()); }
} else {
setPlaylist(urls); QStringList PlaylistManager::autoLoadFilterSuffix() const
return index(0); {
} return m_autoLoadSuffix;
} }
QModelIndex PlaylistModel::loadPlaylist(const QUrl &url) void PlaylistManager::setAutoLoadFilterSuffix(const QStringList & nameFilters)
{ {
QFileInfo info(url.toLocalFile()); m_autoLoadSuffix = nameFilters;
QDir dir(info.path()); }
QString && currentFileName = info.fileName();
void PlaylistManager::clear()
if (dir.path() == m_currentDir) { {
int idx = indexOf(url); m_currentIndex = -1;
return idx == -1 ? appendToPlaylist(url) : index(idx); m_playlist.clear();
} }
QStringList entryList = dir.entryList( void PlaylistManager::setPlaylist(const QList<QUrl> &urls)
m_autoLoadSuffixes, {
QDir::Files | QDir::NoSymLinks, QDir::NoSort); m_playlist = urls;
}
QCollator collator;
collator.setNumericMode(true); void PlaylistManager::setCurrentFile(const QString & filePath)
{
std::sort(entryList.begin(), entryList.end(), collator); QFileInfo info(filePath);
QDir dir(info.path());
QList<QUrl> playlist; QString && currentFileName = info.fileName();
int idx = -1; switch (playlistType()) {
for (int i = 0; i < entryList.count(); i++) { case PL_SAMEFOLDER: {
const QString & fileName = entryList.at(i); if (dir.path() == m_currentDir) {
const QString & oneEntry = dir.absoluteFilePath(fileName); int index = indexOf(filePath);
const QUrl & url = QUrl::fromLocalFile(oneEntry); m_currentIndex = index == -1 ? appendFile(filePath) : index;
playlist.append(url); } else {
if (fileName == currentFileName) { QStringList entryList = dir.entryList(
idx = i; m_autoLoadSuffix,
} QDir::Files | QDir::NoSymLinks, QDir::NoSort);
}
if (idx == -1) { QCollator collator;
idx = playlist.count(); collator.setNumericMode(true);
playlist.append(url);
} std::sort(entryList.begin(), entryList.end(), collator);
m_currentDir = dir.path();
clear();
setPlaylist(playlist);
int index = -1;
return index(idx); for (int i = 0; i < entryList.count(); i++) {
} const QString & fileName = entryList.at(i);
const QString & oneEntry = dir.absoluteFilePath(fileName);
QModelIndex PlaylistModel::appendToPlaylist(const QUrl &url) const QUrl & url = QUrl::fromLocalFile(oneEntry);
{ m_playlist.append(url);
const int lastIndex = rowCount(); if (fileName == currentFileName) {
beginInsertRows(QModelIndex(), lastIndex, lastIndex); index = i;
m_playlist.append(url); }
endInsertRows(); }
return index(lastIndex); m_currentIndex = index == -1 ? appendFile(filePath) : index;
} m_currentDir = dir.path();
}
bool PlaylistModel::removeAt(int index) break;
{ }
if (index < 0 || index >= rowCount()) return false; case PL_USERPLAYLIST:{
beginRemoveRows(QModelIndex(), index, index); int index = indexOf(filePath);
m_playlist.removeAt(index); m_currentIndex = index == -1 ? appendFile(filePath) : index;
endRemoveRows(); break;
return true; }
} default:
break;
int PlaylistModel::indexOf(const QUrl &url) const }
{
return m_playlist.indexOf(url); emit currentIndexChanged(m_currentIndex);
} emit loaded(m_playlist.count());
}
QUrl PlaylistModel::urlByIndex(int index) const
{ void PlaylistManager::setCurrentIndex(int index)
return m_playlist.value(index); {
} if (index < 0 || index >= m_playlist.count()) return;
m_currentIndex = index;
QStringList PlaylistModel::autoLoadFilterSuffixes() const emit currentIndexChanged(m_currentIndex);
{ }
return m_autoLoadSuffixes;
} int PlaylistManager::appendFile(const QString &filePath)
{
QHash<int, QByteArray> PlaylistModel::roleNames() const int index = m_playlist.length();
{ m_playlist.append(QUrl::fromLocalFile(filePath));
QHash<int, QByteArray> result = QAbstractListModel::roleNames();
result.insert(UrlRole, "url"); return index;
return result; }
}
int PlaylistManager::indexOf(const QString &filePath)
int PlaylistModel::rowCount(const QModelIndex &parent) const {
{ const QUrl & url = QUrl::fromLocalFile(filePath);
return m_playlist.count(); return m_playlist.indexOf(url);
} }
QVariant PlaylistModel::data(const QModelIndex &index, int role) const int PlaylistManager::count() const
{ {
if (!index.isValid()) return {}; return m_playlist.count();
}
switch (role) {
case Qt::DisplayRole: std::tuple<int, QString> PlaylistManager::previousFile() const
return m_playlist.at(index.row()).fileName(); {
case UrlRole: int count = m_playlist.count();
return m_playlist.at(index.row()); if (count == 0) return std::make_tuple(-1, QString());
}
int index = m_currentIndex - 1 < 0 ? count - 1 : m_currentIndex - 1;
return {}; return std::make_tuple(index, m_playlist.at(index).toLocalFile());
} }
PlaylistManager::PlaylistManager(QObject *parent) std::tuple<int, QString> PlaylistManager::nextFile() const
: QObject(parent) {
{ int count = m_playlist.count();
connect(&m_model, &PlaylistModel::rowsRemoved, this, if (count == 0) return std::make_tuple(-1, QString());
[this](const QModelIndex &, int, int) {
if (m_model.rowCount() <= m_currentIndex) { int index = m_currentIndex + 1 == count ? 0 : m_currentIndex + 1;
setProperty("currentIndex", m_currentIndex - 1); return std::make_tuple(index, m_playlist.at(index).toLocalFile());
} }
});
std::tuple<int, QString> PlaylistManager::currentFile() const
auto onRowCountChanged = [this](){ {
emit totalCountChanged(m_model.rowCount()); if (m_playlist.count() == 0) return std::make_tuple(-1, QString());
};
return std::make_tuple(m_currentIndex, m_playlist.at(m_currentIndex).toLocalFile());
connect(&m_model, &PlaylistModel::rowsInserted, this, onRowCountChanged); }
connect(&m_model, &PlaylistModel::rowsRemoved, this, onRowCountChanged);
connect(&m_model, &PlaylistModel::modelReset, this, onRowCountChanged); std::tuple<int, QUrl> PlaylistManager::currentFileUrl() const
} {
if (m_playlist.count() == 0) return std::make_tuple(-1, QUrl());
PlaylistManager::~PlaylistManager()
{ return std::make_tuple(m_currentIndex, m_playlist.at(m_currentIndex));
}
}
QList<QUrl> PlaylistManager::convertToUrlList(const QStringList &files)
PlaylistModel *PlaylistManager::model() {
{ QList<QUrl> urlList;
return &m_model; for (const QString & str : qAsConst(files)) {
} QUrl url = QUrl::fromLocalFile(str);
if (url.isValid()) {
void PlaylistManager::setPlaylist(const QList<QUrl> &urls) urlList.append(url);
{ }
m_model.setPlaylist(urls); }
}
return urlList;
QModelIndex PlaylistManager::loadPlaylist(const QList<QUrl> &urls) }
{
QModelIndex idx = m_model.loadPlaylist(urls);
setProperty("currentIndex", idx.row());
return idx;
}
QModelIndex PlaylistManager::loadPlaylist(const QUrl &url)
{
QModelIndex idx = m_model.loadPlaylist(url);
setProperty("currentIndex", idx.row());
return idx;
}
QModelIndex PlaylistManager::loadM3U8Playlist(const QUrl &url)
{
QFile file(url.toLocalFile());
if (file.open(QIODevice::ReadOnly | QIODevice::Text)) {
QList<QUrl> urls;
while (!file.atEnd()) {
QString line = file.readLine();
if (line.startsWith('#')) {
continue;
}
QFileInfo fileInfo(file);
QUrl item = QUrl::fromUserInput(line, fileInfo.absolutePath());
urls.append(item);
}
return loadPlaylist(urls);
} else {
return {};
}
}
int PlaylistManager::totalCount() const
{
return m_model.rowCount();
}
QModelIndex PlaylistManager::previousIndex() const
{
int count = totalCount();
if (count == 0) return {};
return m_model.index(isFirstIndex() ? count - 1 : m_currentIndex - 1);
}
QModelIndex PlaylistManager::nextIndex() const
{
int count = totalCount();
if (count == 0) return {};
return m_model.index(isLastIndex() ? 0 : m_currentIndex + 1);
}
QModelIndex PlaylistManager::curIndex() const
{
return m_model.index(m_currentIndex);
}
bool PlaylistManager::isFirstIndex() const
{
return m_currentIndex == 0;
}
bool PlaylistManager::isLastIndex() const
{
return m_currentIndex + 1 == totalCount();
}
void PlaylistManager::setCurrentIndex(const QModelIndex &index)
{
if (index.isValid() && index.row() >= 0 && index.row() < totalCount()) {
setProperty("currentIndex", index.row());
}
}
QUrl PlaylistManager::urlByIndex(const QModelIndex &index)
{
return m_model.urlByIndex(index.row());
}
QString PlaylistManager::localFileByIndex(const QModelIndex &index)
{
return urlByIndex(index).toLocalFile();
}
bool PlaylistManager::removeAt(const QModelIndex &index)
{
return m_model.removeAt(index.row());
}
void PlaylistManager::setAutoLoadFilterSuffixes(const QStringList &nameFilters)
{
m_model.setProperty("autoLoadFilterSuffixes", nameFilters);
}
QList<QUrl> PlaylistManager::convertToUrlList(const QStringList &files)
{
QList<QUrl> urlList;
for (const QString & str : std::as_const(files)) {
QUrl url = QUrl::fromLocalFile(str);
if (url.isValid()) {
urlList.append(url);
}
}
return urlList;
}

View File

@ -1,88 +1,57 @@
// SPDX-FileCopyrightText: 2025 Gary Wang <git@blumia.net> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#pragma once #pragma once
#include <QUrl> #include <QObject>
#include <QAbstractListModel>
class PlaylistManager : public QObject
class PlaylistModel : public QAbstractListModel {
{ Q_OBJECT
Q_OBJECT public:
public: Q_PROPERTY(int currentIndex MEMBER m_currentIndex NOTIFY currentIndexChanged)
enum PlaylistRole {
UrlRole = Qt::UserRole enum PlaylistType {
}; PL_USERPLAYLIST, // Regular playlist, managed by user.
Q_ENUM(PlaylistRole) PL_SAMEFOLDER // PlaylistManager managed playlist, loaded from files from same folder.
Q_PROPERTY(QStringList autoLoadFilterSuffixes MEMBER m_autoLoadSuffixes NOTIFY autoLoadFilterSuffixesChanged) };
explicit PlaylistModel(QObject *parent = nullptr); explicit PlaylistManager(PlaylistType type = PL_USERPLAYLIST, QObject *parent = nullptr);
~PlaylistModel() override; ~PlaylistManager();
void setPlaylist(const QList<QUrl> & urls); void setPlaylistType(PlaylistType type);
QModelIndex loadPlaylist(const QList<QUrl> & urls); PlaylistType playlistType() const;
QModelIndex loadPlaylist(const QUrl & url);
QModelIndex appendToPlaylist(const QUrl & url); QStringList autoLoadFilterSuffix() const;
bool removeAt(int index); void setAutoLoadFilterSuffix(const QStringList &nameFilters);
int indexOf(const QUrl & url) const;
QUrl urlByIndex(int index) const; void clear();
QStringList autoLoadFilterSuffixes() const;
void setPlaylist(const QList<QUrl> & urls);
QHash<int, QByteArray> roleNames() const override; void setCurrentFile(const QString & filePath);
int rowCount(const QModelIndex &parent = QModelIndex()) const override; void setCurrentIndex(int index);
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override; int appendFile(const QString & filePath);
int indexOf(const QString & filePath);
signals:
void autoLoadFilterSuffixesChanged(QStringList suffixes); int count() const;
private: std::tuple<int, QString> previousFile() const;
// model data std::tuple<int, QString> nextFile() const;
QList<QUrl> m_playlist; std::tuple<int, QString> currentFile() const;
// properties std::tuple<int, QUrl> currentFileUrl() const;
QStringList m_autoLoadSuffixes = {};
// internal static QList<QUrl> convertToUrlList(const QStringList & files);
QString m_currentDir;
}; signals:
void loaded(int length);
class PlaylistManager : public QObject void currentIndexChanged(int index);
{
Q_OBJECT private:
public: QList<QUrl> m_playlist;
Q_PROPERTY(int currentIndex MEMBER m_currentIndex NOTIFY currentIndexChanged) PlaylistType m_type;
Q_PROPERTY(QStringList autoLoadFilterSuffixes WRITE setAutoLoadFilterSuffixes) QString m_currentDir;
Q_PROPERTY(PlaylistModel * model READ model CONSTANT) int m_currentIndex = -1;
QStringList m_autoLoadSuffix = {};
explicit PlaylistManager(QObject *parent = nullptr); };
~PlaylistManager();
PlaylistModel * model();
void setPlaylist(const QList<QUrl> & url);
Q_INVOKABLE QModelIndex loadPlaylist(const QList<QUrl> & urls);
Q_INVOKABLE QModelIndex loadPlaylist(const QUrl & url);
Q_INVOKABLE QModelIndex loadM3U8Playlist(const QUrl & url);
int totalCount() const;
QModelIndex previousIndex() const;
QModelIndex nextIndex() const;
QModelIndex curIndex() const;
bool isFirstIndex() const;
bool isLastIndex() const;
void setCurrentIndex(const QModelIndex & index);
QUrl urlByIndex(const QModelIndex & index);
QString localFileByIndex(const QModelIndex & index);
bool removeAt(const QModelIndex & index);
void setAutoLoadFilterSuffixes(const QStringList &nameFilters);
static QList<QUrl> convertToUrlList(const QStringList & files);
signals:
void currentIndexChanged(int index);
void totalCountChanged(int count);
private:
int m_currentIndex = -1;
PlaylistModel m_model;
};

View File

@ -1,226 +1,147 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "settings.h" #include "settings.h"
#include <QApplication> #include <QApplication>
#include <QStandardPaths> #include <QStandardPaths>
#include <QDebug> #include <QDebug>
#include <QDir> #include <QDir>
#include <QAction> #include <QMetaEnum>
#include <QWidget>
#include <QKeySequence> namespace QEnumHelper
#include <QMetaEnum> {
template <typename E>
namespace QEnumHelper E fromString(const QString &text, const E defaultValue)
{ {
template <typename E> bool ok;
E fromString(const QString &text, const E defaultValue) E result = static_cast<E>(QMetaEnum::fromType<E>().keyToValue(text.toUtf8(), &ok));
{ if (!ok) {
bool ok; return defaultValue;
E result = static_cast<E>(QMetaEnum::fromType<E>().keyToValue(text.toUtf8(), &ok)); }
if (!ok) { return result;
return defaultValue; }
}
return result; template <typename E>
} QString toString(E value)
{
template <typename E> const int intValue = static_cast<int>(value);
QString toString(E value) return QString::fromUtf8(QMetaEnum::fromType<E>().valueToKey(intValue));
{ }
const int intValue = static_cast<int>(value); }
return QString::fromUtf8(QMetaEnum::fromType<E>().valueToKey(intValue));
} Settings *Settings::m_settings_instance = nullptr;
}
Settings *Settings::instance()
Settings *Settings::m_settings_instance = nullptr; {
if (!m_settings_instance) {
Settings *Settings::instance() m_settings_instance = new Settings;
{ }
if (!m_settings_instance) {
m_settings_instance = new Settings; return m_settings_instance;
} }
return m_settings_instance; bool Settings::stayOnTop()
} {
return m_qsettings->value("stay_on_top", true).toBool();
bool Settings::stayOnTop() const }
{
return m_qsettings->value("stay_on_top", true).toBool(); Settings::DoubleClickBehavior Settings::doubleClickBehavior() const
} {
QString result = m_qsettings->value("double_click_behavior", "Close").toString();
bool Settings::useBuiltInCloseAnimation() const
{ return QEnumHelper::fromString<DoubleClickBehavior>(result, DoubleClickBehavior::Close);
return m_qsettings->value("use_built_in_close_animation", true).toBool(); }
}
Settings::MouseWheelBehavior Settings::mouseWheelBehavior() const
bool Settings::useLightCheckerboard() const {
{ QString result = m_qsettings->value("mouse_wheel_behavior", "Zoom").toString();
return m_qsettings->value("use_light_checkerboard", false).toBool();
} return QEnumHelper::fromString<MouseWheelBehavior>(result, MouseWheelBehavior::Zoom);
}
bool Settings::loopGallery() const
{ Settings::WindowSizeBehavior Settings::initWindowSizeBehavior() const
return m_qsettings->value("loop_gallery", true).toBool(); {
} QString result = m_qsettings->value("init_window_size_behavior", "Auto").toString();
Settings::DoubleClickBehavior Settings::doubleClickBehavior() const return QEnumHelper::fromString<WindowSizeBehavior>(result, WindowSizeBehavior::Auto);
{ }
QString result = m_qsettings->value("double_click_behavior", "Close").toString();
Qt::HighDpiScaleFactorRoundingPolicy Settings::hiDpiScaleFactorBehavior() const
return QEnumHelper::fromString<DoubleClickBehavior>(result, DoubleClickBehavior::Close); {
} QString result = m_qsettings->value("hidpi_scale_factor_behavior", "PassThrough").toString();
Settings::MouseWheelBehavior Settings::mouseWheelBehavior() const return QEnumHelper::fromString<Qt::HighDpiScaleFactorRoundingPolicy>(result, Qt::HighDpiScaleFactorRoundingPolicy::PassThrough);
{ }
QString result = m_qsettings->value("mouse_wheel_behavior", "Zoom").toString();
void Settings::setStayOnTop(bool on)
return QEnumHelper::fromString<MouseWheelBehavior>(result, MouseWheelBehavior::Zoom); {
} m_qsettings->setValue("stay_on_top", on);
m_qsettings->sync();
Settings::WindowSizeBehavior Settings::initWindowSizeBehavior() const }
{
QString result = m_qsettings->value("init_window_size_behavior", "Auto").toString(); void Settings::setDoubleClickBehavior(DoubleClickBehavior dcb)
{
return QEnumHelper::fromString<WindowSizeBehavior>(result, WindowSizeBehavior::Auto); m_qsettings->setValue("double_click_behavior", QEnumHelper::toString(dcb));
} m_qsettings->sync();
}
Qt::HighDpiScaleFactorRoundingPolicy Settings::hiDpiScaleFactorBehavior() const
{ void Settings::setMouseWheelBehavior(MouseWheelBehavior mwb)
QString result = m_qsettings->value("hidpi_scale_factor_behavior", "PassThrough").toString(); {
m_qsettings->setValue("mouse_wheel_behavior", QEnumHelper::toString(mwb));
return QEnumHelper::fromString<Qt::HighDpiScaleFactorRoundingPolicy>(result, Qt::HighDpiScaleFactorRoundingPolicy::PassThrough); m_qsettings->sync();
} }
void Settings::setStayOnTop(bool on) void Settings::setInitWindowSizeBehavior(WindowSizeBehavior wsb)
{ {
m_qsettings->setValue("stay_on_top", on); m_qsettings->setValue("init_window_size_behavior", QEnumHelper::toString(wsb));
m_qsettings->sync(); m_qsettings->sync();
} }
void Settings::setUseBuiltInCloseAnimation(bool on) void Settings::setHiDpiScaleFactorBehavior(Qt::HighDpiScaleFactorRoundingPolicy hidpi)
{ {
m_qsettings->setValue("use_built_in_close_animation", on); m_qsettings->setValue("hidpi_scale_factor_behavior", QEnumHelper::toString(hidpi));
m_qsettings->sync(); m_qsettings->sync();
} }
void Settings::setUseLightCheckerboard(bool light) #if defined(FLAG_PORTABLE_MODE_SUPPORT) && defined(Q_OS_WIN)
{ #include <windows.h>
m_qsettings->setValue("use_light_checkerboard", light); // QCoreApplication::applicationDirPath() parses the "applicationDirPath" from arg0, which...
m_qsettings->sync(); // 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
void Settings::setLoopGallery(bool on) // see also: https://stackoverflow.com/questions/383973/is-args0-guaranteed-to-be-the-path-of-execution
{ // This function is here mainly for #1.
m_qsettings->setValue("loop_gallery", on); QString getApplicationDirPath()
m_qsettings->sync(); {
} WCHAR buffer[MAX_PATH];
GetModuleFileNameW(NULL, buffer, MAX_PATH);
void Settings::setDoubleClickBehavior(DoubleClickBehavior dcb) QString appPath = QString::fromWCharArray(buffer);
{
m_qsettings->setValue("double_click_behavior", QEnumHelper::toString(dcb)); return appPath.left(appPath.lastIndexOf('\\'));
m_qsettings->sync(); }
} #endif // defined(FLAG_PORTABLE_MODE_SUPPORT) && defined(Q_OS_WIN)
void Settings::setMouseWheelBehavior(MouseWheelBehavior mwb) Settings::Settings()
{ : QObject(qApp)
m_qsettings->setValue("mouse_wheel_behavior", QEnumHelper::toString(mwb)); {
m_qsettings->sync(); QString configPath;
}
#if defined(FLAG_PORTABLE_MODE_SUPPORT) && defined(Q_OS_WIN)
void Settings::setInitWindowSizeBehavior(WindowSizeBehavior wsb) QString portableConfigDirPath = QDir(getApplicationDirPath()).absoluteFilePath("data");
{ QFileInfo portableConfigDirInfo(portableConfigDirPath);
m_qsettings->setValue("init_window_size_behavior", QEnumHelper::toString(wsb)); if (portableConfigDirInfo.exists() && portableConfigDirInfo.isDir() && portableConfigDirInfo.isWritable()) {
m_qsettings->sync(); // we can use it.
} configPath = portableConfigDirPath;
}
void Settings::setHiDpiScaleFactorBehavior(Qt::HighDpiScaleFactorRoundingPolicy hidpi) #endif // defined(FLAG_PORTABLE_MODE_SUPPORT) && defined(Q_OS_WIN)
{
m_qsettings->setValue("hidpi_scale_factor_behavior", QEnumHelper::toString(hidpi)); if (configPath.isEmpty()) {
m_qsettings->sync(); // %LOCALAPPDATA%\<APPNAME> under Windows, ~/.config/<APPNAME> under Linux.
} configPath = QStandardPaths::writableLocation(QStandardPaths::AppConfigLocation);
}
void Settings::applyUserShortcuts(QWidget *widget)
{ m_qsettings = new QSettings(QDir(configPath).absoluteFilePath("config.ini"), QSettings::IniFormat, this);
m_qsettings->beginGroup("shortcuts"); }
const QStringList shortcutNames = m_qsettings->allKeys();
for (const QString & name : shortcutNames) {
QList<QKeySequence> shortcuts = m_qsettings->value(name).value<QList<QKeySequence>>();
setShortcutsForAction(widget, name, shortcuts, false);
}
m_qsettings->endGroup();
}
bool Settings::setShortcutsForAction(QWidget *widget, const QString &objectName,
QList<QKeySequence> shortcuts, bool writeConfig)
{
QAction * targetAction = nullptr;
for (QAction * action : widget->actions()) {
if (action->objectName() == objectName) {
targetAction = action;
} else {
for (const QKeySequence & shortcut : std::as_const(shortcuts)) {
if (action->shortcuts().contains(shortcut)) {
return false;
}
}
}
}
if (targetAction) {
targetAction->setShortcuts(shortcuts);
}
if (targetAction && writeConfig) {
m_qsettings->beginGroup("shortcuts");
m_qsettings->setValue(objectName, QVariant::fromValue(shortcuts));
m_qsettings->endGroup();
m_qsettings->sync();
}
return true;
}
#if defined(FLAG_PORTABLE_MODE_SUPPORT) && defined(Q_OS_WIN)
#include <windows.h>
// 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()) {
// Should be %LOCALAPPDATA%\<APPNAME> under Windows, ~/.config/<APPNAME> under Linux.
configPath = QStandardPaths::writableLocation(QStandardPaths::AppConfigLocation);
}
m_qsettings = new QSettings(QDir(configPath).absoluteFilePath("config.ini"), QSettings::IniFormat, this);
qRegisterMetaType<QList<QKeySequence>>();
}

View File

@ -1,69 +1,58 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#pragma once #pragma once
#include <QObject> #include <QObject>
#include <QSettings> #include <QSettings>
class Settings : public QObject class Settings : public QObject
{ {
Q_OBJECT Q_OBJECT
public: public:
enum DoubleClickBehavior { enum DoubleClickBehavior {
Ignore, Ignore,
Close, Close,
Maximize, Maximize,
FullScreen, };
}; Q_ENUM(DoubleClickBehavior)
Q_ENUM(DoubleClickBehavior)
enum MouseWheelBehavior {
enum MouseWheelBehavior { Zoom,
Zoom, Switch,
Switch, };
}; Q_ENUM(MouseWheelBehavior)
Q_ENUM(MouseWheelBehavior)
enum WindowSizeBehavior {
enum WindowSizeBehavior { Auto,
Auto, Maximized,
Maximized, };
Windowed, Q_ENUM(WindowSizeBehavior)
};
Q_ENUM(WindowSizeBehavior) static Settings *instance();
static Settings *instance(); bool stayOnTop();
DoubleClickBehavior doubleClickBehavior() const;
bool stayOnTop() const; MouseWheelBehavior mouseWheelBehavior() const;
bool useBuiltInCloseAnimation() const; WindowSizeBehavior initWindowSizeBehavior() const;
bool useLightCheckerboard() const; Qt::HighDpiScaleFactorRoundingPolicy hiDpiScaleFactorBehavior() const;
bool loopGallery() const;
DoubleClickBehavior doubleClickBehavior() const; void setStayOnTop(bool on);
MouseWheelBehavior mouseWheelBehavior() const; void setDoubleClickBehavior(DoubleClickBehavior dcb);
WindowSizeBehavior initWindowSizeBehavior() const; void setMouseWheelBehavior(MouseWheelBehavior mwb);
Qt::HighDpiScaleFactorRoundingPolicy hiDpiScaleFactorBehavior() const; void setInitWindowSizeBehavior(WindowSizeBehavior wsb);
void setHiDpiScaleFactorBehavior(Qt::HighDpiScaleFactorRoundingPolicy hidpi);
void setStayOnTop(bool on);
void setUseBuiltInCloseAnimation(bool on); private:
void setUseLightCheckerboard(bool light); Settings();
void setLoopGallery(bool on);
void setDoubleClickBehavior(DoubleClickBehavior dcb); static Settings *m_settings_instance;
void setMouseWheelBehavior(MouseWheelBehavior mwb);
void setInitWindowSizeBehavior(WindowSizeBehavior wsb); QSettings *m_qsettings;
void setHiDpiScaleFactorBehavior(Qt::HighDpiScaleFactorRoundingPolicy hidpi);
signals:
void applyUserShortcuts(QWidget * widget);
bool setShortcutsForAction(QWidget * widget, const QString & objectName, public slots:
QList<QKeySequence> shortcuts, bool writeConfig = true); };
private:
Settings();
static Settings *m_settings_instance;
QSettings *m_qsettings;
signals:
public slots:
};

View File

@ -1,200 +1,121 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "settingsdialog.h" #include "settingsdialog.h"
#include "settings.h" #include "settings.h"
#include "shortcutedit.h"
#include <QCheckBox>
#include <QAction> #include <QComboBox>
#include <QCheckBox> #include <QFormLayout>
#include <QComboBox> #include <QStringListModel>
#include <QFormLayout>
#include <QKeySequenceEdit> SettingsDialog::SettingsDialog(QWidget *parent)
#include <QScrollArea> : QDialog(parent)
#include <QSplitter> , m_stayOnTop(new QCheckBox)
#include <QStringListModel> , m_doubleClickBehavior(new QComboBox)
#include <QMessageBox> , m_mouseWheelBehavior(new QComboBox)
, m_initWindowSizeBehavior(new QComboBox)
SettingsDialog::SettingsDialog(QWidget *parent) , m_hiDpiRoundingPolicyBehavior(new QComboBox)
: QDialog(parent) {
, m_stayOnTop(new QCheckBox) this->setWindowTitle(tr("Settings"));
, m_useBuiltInCloseAnimation(new QCheckBox)
, m_useLightCheckerboard(new QCheckBox) QFormLayout * settingsForm = new QFormLayout(this);
, m_loopGallery(new QCheckBox)
, m_doubleClickBehavior(new QComboBox) static QList< QPair<Settings::DoubleClickBehavior, QString> > _dc_options {
, m_mouseWheelBehavior(new QComboBox) { Settings::DoubleClickBehavior::Ignore, tr("Do nothing") },
, m_initWindowSizeBehavior(new QComboBox) { Settings::DoubleClickBehavior::Close, tr("Close the window") },
, m_hiDpiRoundingPolicyBehavior(new QComboBox) { Settings::DoubleClickBehavior::Maximize, tr("Toggle maximize") }
{ };
this->setWindowTitle(tr("Settings"));
static QList< QPair<Settings::MouseWheelBehavior, QString> > _mw_options {
QHBoxLayout * mainLayout = new QHBoxLayout(this); { Settings::MouseWheelBehavior::Zoom, tr("Zoom in and out") },
QTabWidget * settingsTabs = new QTabWidget(this); { Settings::MouseWheelBehavior::Switch, tr("View next or previous item") }
mainLayout->addWidget(settingsTabs); };
QWidget * settingsFormHolder = new QWidget; static QList< QPair<Settings::WindowSizeBehavior, QString> > _iws_options {
QFormLayout * settingsForm = new QFormLayout(settingsFormHolder); { Settings::WindowSizeBehavior::Auto, tr("Auto size") },
settingsTabs->addTab(settingsFormHolder, tr("Options")); { Settings::WindowSizeBehavior::Maximized, tr("Maximized") }
};
QSplitter * shortcutEditorSplitter = new QSplitter;
shortcutEditorSplitter->setOrientation(Qt::Vertical); static QList< QPair<Qt::HighDpiScaleFactorRoundingPolicy, QString> > _hidpi_options {
shortcutEditorSplitter->setChildrenCollapsible(false); { Qt::HighDpiScaleFactorRoundingPolicy::Round, tr("Round (Integer scaling)", "This option means round up for .5 and above") },
QScrollArea * shortcutScrollArea = new QScrollArea; { Qt::HighDpiScaleFactorRoundingPolicy::Ceil, tr("Ceil (Integer scaling)", "This option means always round up") },
shortcutEditorSplitter->addWidget(shortcutScrollArea); { Qt::HighDpiScaleFactorRoundingPolicy::Floor, tr("Floor (Integer scaling)", "This option means always round down") },
shortcutScrollArea->setWidgetResizable(true); { Qt::HighDpiScaleFactorRoundingPolicy::PassThrough, tr("Follow system (Fractional scaling)", "This option means don't round") }
shortcutScrollArea->setMinimumHeight(200); };
QWidget * shortcutsFormHolder = new QWidget;
QFormLayout * shortcutsForm = new QFormLayout(shortcutsFormHolder); QStringList dcbDropDown;
shortcutScrollArea->setWidget(shortcutsFormHolder); for (const QPair<Settings::DoubleClickBehavior, QString> & dcOption : _dc_options) {
settingsTabs->addTab(shortcutEditorSplitter, tr("Shortcuts")); dcbDropDown.append(dcOption.second);
}
for (const QAction * action : parent->actions()) {
ShortcutEdit * shortcutEdit = new ShortcutEdit; QStringList mwbDropDown;
shortcutEdit->setObjectName(QLatin1String("shortcut_") + action->objectName()); for (const QPair<Settings::MouseWheelBehavior, QString> & mwOption : _mw_options) {
shortcutEdit->setShortcuts(action->shortcuts()); mwbDropDown.append(mwOption.second);
shortcutsForm->addRow(action->text(), shortcutEdit); }
connect(shortcutEdit, &ShortcutEdit::editButtonClicked, this, [=](){
if (shortcutEditorSplitter->count() == 1) shortcutEditorSplitter->addWidget(new QWidget); QStringList iwsbDropDown;
ShortcutEditor * shortcutEditor = new ShortcutEditor(shortcutEdit); for (const QPair<Settings::WindowSizeBehavior, QString> & iwsOption : _iws_options) {
shortcutEditor->setDescription(tr("Editing shortcuts for action \"%1\":").arg(action->text())); iwsbDropDown.append(iwsOption.second);
QWidget * oldEditor = shortcutEditorSplitter->replaceWidget(1, shortcutEditor); }
shortcutEditorSplitter->setSizes({shortcutEditorSplitter->height(), 1});
oldEditor->deleteLater(); QStringList hidpiDropDown;
}); for (const QPair<Qt::HighDpiScaleFactorRoundingPolicy, QString> & hidpiOption : _hidpi_options) {
connect(shortcutEdit, &ShortcutEdit::applyShortcutsRequested, this, [=](QList<QKeySequence> newShortcuts){ hidpiDropDown.append(hidpiOption.second);
bool succ = Settings::instance()->setShortcutsForAction(parent, shortcutEdit->objectName().mid(9), }
newShortcuts);
if (!succ) { settingsForm->addRow(tr("Stay on top when start-up"), m_stayOnTop);
QMessageBox::warning(this, tr("Failed to set shortcuts"), settingsForm->addRow(tr("Double-click behavior"), m_doubleClickBehavior);
tr("Please check if shortcuts are duplicated with existing shortcuts.")); settingsForm->addRow(tr("Mouse wheel behavior"), m_mouseWheelBehavior);
} settingsForm->addRow(tr("Default window size"), m_initWindowSizeBehavior);
shortcutEdit->setShortcuts(action->shortcuts()); settingsForm->addRow(tr("HiDPI scale factor rounding policy"), m_hiDpiRoundingPolicyBehavior);
});
} m_stayOnTop->setChecked(Settings::instance()->stayOnTop());
m_doubleClickBehavior->setModel(new QStringListModel(dcbDropDown));
static QList< QPair<Settings::DoubleClickBehavior, QString> > _dc_options { Settings::DoubleClickBehavior dcb = Settings::instance()->doubleClickBehavior();
{ Settings::DoubleClickBehavior::Ignore, tr("Do nothing") }, m_doubleClickBehavior->setCurrentIndex(static_cast<int>(dcb));
{ Settings::DoubleClickBehavior::Close, tr("Close the window") }, m_mouseWheelBehavior->setModel(new QStringListModel(mwbDropDown));
{ Settings::DoubleClickBehavior::Maximize, tr("Toggle maximize") }, Settings::MouseWheelBehavior mwb = Settings::instance()->mouseWheelBehavior();
{ Settings::DoubleClickBehavior::FullScreen, tr("Toggle fullscreen") } m_mouseWheelBehavior->setCurrentIndex(static_cast<int>(mwb));
}; m_initWindowSizeBehavior->setModel(new QStringListModel(iwsbDropDown));
Settings::WindowSizeBehavior iwsb = Settings::instance()->initWindowSizeBehavior();
static QList< QPair<Settings::MouseWheelBehavior, QString> > _mw_options { m_initWindowSizeBehavior->setCurrentIndex(static_cast<int>(iwsb));
{ Settings::MouseWheelBehavior::Zoom, tr("Zoom in and out") }, m_hiDpiRoundingPolicyBehavior->setModel(new QStringListModel(hidpiDropDown));
{ Settings::MouseWheelBehavior::Switch, tr("View next or previous item") } Qt::HighDpiScaleFactorRoundingPolicy hidpi = Settings::instance()->hiDpiScaleFactorBehavior();
}; for (int i = 0; i < _hidpi_options.count(); i++) {
if (_hidpi_options.at(i).first == hidpi) {
static QList< QPair<Settings::WindowSizeBehavior, QString> > _iws_options { m_hiDpiRoundingPolicyBehavior->setCurrentIndex(i);
{ Settings::WindowSizeBehavior::Auto, tr("Auto size") }, break;
{ Settings::WindowSizeBehavior::Maximized, tr("Maximized") }, }
{ Settings::WindowSizeBehavior::Windowed, tr("Windowed") } }
};
connect(m_stayOnTop, &QCheckBox::stateChanged, this, [ = ](int state){
static QList< QPair<Qt::HighDpiScaleFactorRoundingPolicy, QString> > _hidpi_options { Settings::instance()->setStayOnTop(state == Qt::Checked);
{ 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") }, connect(m_doubleClickBehavior, QOverload<int>::of(&QComboBox::currentIndexChanged), this, [ = ](int index){
{ Qt::HighDpiScaleFactorRoundingPolicy::PassThrough, tr("Follow system (Fractional scaling)", "This option means don't round") } Settings::instance()->setDoubleClickBehavior(_dc_options.at(index).first);
}; });
QStringList dcbDropDown; connect(m_mouseWheelBehavior, QOverload<int>::of(&QComboBox::currentIndexChanged), this, [ = ](int index){
for (const QPair<Settings::DoubleClickBehavior, QString> & dcOption : _dc_options) { Settings::instance()->setMouseWheelBehavior(_mw_options.at(index).first);
dcbDropDown.append(dcOption.second); });
}
connect(m_initWindowSizeBehavior, QOverload<int>::of(&QComboBox::currentIndexChanged), this, [ = ](int index){
QStringList mwbDropDown; Settings::instance()->setInitWindowSizeBehavior(_iws_options.at(index).first);
for (const QPair<Settings::MouseWheelBehavior, QString> & mwOption : _mw_options) { });
mwbDropDown.append(mwOption.second);
} connect(m_hiDpiRoundingPolicyBehavior, QOverload<int>::of(&QComboBox::currentIndexChanged), this, [ = ](int index){
Settings::instance()->setHiDpiScaleFactorBehavior(_hidpi_options.at(index).first);
QStringList iwsbDropDown; });
for (const QPair<Settings::WindowSizeBehavior, QString> & iwsOption : _iws_options) {
iwsbDropDown.append(iwsOption.second); adjustSize();
} setWindowFlag(Qt::WindowContextHelpButtonHint, false);
}
QStringList hidpiDropDown;
for (const QPair<Qt::HighDpiScaleFactorRoundingPolicy, QString> & hidpiOption : _hidpi_options) { SettingsDialog::~SettingsDialog()
hidpiDropDown.append(hidpiOption.second); {
}
}
settingsForm->addRow(tr("Stay on top when start-up"), m_stayOnTop);
settingsForm->addRow(tr("Use built-in close window animation"), m_useBuiltInCloseAnimation);
settingsForm->addRow(tr("Use light-color checkerboard"), m_useLightCheckerboard);
settingsForm->addRow(tr("Loop the loaded gallery"), m_loopGallery);
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_useBuiltInCloseAnimation->setChecked(Settings::instance()->useBuiltInCloseAnimation());
m_useLightCheckerboard->setChecked(Settings::instance()->useLightCheckerboard());
m_loopGallery->setChecked(Settings::instance()->loopGallery());
m_doubleClickBehavior->setModel(new QStringListModel(dcbDropDown));
Settings::DoubleClickBehavior dcb = Settings::instance()->doubleClickBehavior();
m_doubleClickBehavior->setCurrentIndex(static_cast<int>(dcb));
m_mouseWheelBehavior->setModel(new QStringListModel(mwbDropDown));
Settings::MouseWheelBehavior mwb = Settings::instance()->mouseWheelBehavior();
m_mouseWheelBehavior->setCurrentIndex(static_cast<int>(mwb));
m_initWindowSizeBehavior->setModel(new QStringListModel(iwsbDropDown));
Settings::WindowSizeBehavior iwsb = Settings::instance()->initWindowSizeBehavior();
m_initWindowSizeBehavior->setCurrentIndex(static_cast<int>(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;
}
}
#if QT_VERSION >= QT_VERSION_CHECK(6, 7, 0)
# define QCHECKBOX_CHECKSTATECHANGED QCheckBox::checkStateChanged
# define QT_CHECKSTATE Qt::CheckState
#else
# define QCHECKBOX_CHECKSTATECHANGED QCheckBox::stateChanged
# define QT_CHECKSTATE int
#endif // QT_VERSION >= QT_VERSION_CHECK(6, 7, 0)
connect(m_stayOnTop, &QCHECKBOX_CHECKSTATECHANGED, this, [ = ](QT_CHECKSTATE state){
Settings::instance()->setStayOnTop(state == Qt::Checked);
});
connect(m_useBuiltInCloseAnimation, &QCHECKBOX_CHECKSTATECHANGED, this, [ = ](QT_CHECKSTATE state){
Settings::instance()->setUseBuiltInCloseAnimation(state == Qt::Checked);
});
connect(m_useLightCheckerboard, &QCHECKBOX_CHECKSTATECHANGED, this, [ = ](QT_CHECKSTATE state){
Settings::instance()->setUseLightCheckerboard(state == Qt::Checked);
});
connect(m_loopGallery, &QCHECKBOX_CHECKSTATECHANGED, this, [ = ](QT_CHECKSTATE state){
Settings::instance()->setLoopGallery(state == Qt::Checked);
});
connect(m_doubleClickBehavior, QOverload<int>::of(&QComboBox::currentIndexChanged), this, [ = ](int index){
Settings::instance()->setDoubleClickBehavior(_dc_options.at(index).first);
});
connect(m_mouseWheelBehavior, QOverload<int>::of(&QComboBox::currentIndexChanged), this, [ = ](int index){
Settings::instance()->setMouseWheelBehavior(_mw_options.at(index).first);
});
connect(m_initWindowSizeBehavior, QOverload<int>::of(&QComboBox::currentIndexChanged), this, [ = ](int index){
Settings::instance()->setInitWindowSizeBehavior(_iws_options.at(index).first);
});
connect(m_hiDpiRoundingPolicyBehavior, QOverload<int>::of(&QComboBox::currentIndexChanged), this, [ = ](int index){
Settings::instance()->setHiDpiScaleFactorBehavior(_hidpi_options.at(index).first);
});
adjustSize();
setWindowFlag(Qt::WindowContextHelpButtonHint, false);
}
SettingsDialog::~SettingsDialog()
{
}

View File

@ -1,35 +1,32 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#ifndef SETTINGSDIALOG_H #ifndef SETTINGSDIALOG_H
#define SETTINGSDIALOG_H #define SETTINGSDIALOG_H
#include <QObject> #include <QObject>
#include <QDialog> #include <QDialog>
class QCheckBox; class QCheckBox;
class QComboBox; class QComboBox;
class SettingsDialog : public QDialog class SettingsDialog : public QDialog
{ {
Q_OBJECT Q_OBJECT
public: public:
explicit SettingsDialog(QWidget *parent = nullptr); explicit SettingsDialog(QWidget *parent = nullptr);
~SettingsDialog(); ~SettingsDialog();
signals: signals:
public slots: public slots:
private: private:
QCheckBox * m_stayOnTop = nullptr; QCheckBox * m_stayOnTop = nullptr;
QCheckBox * m_useBuiltInCloseAnimation = nullptr; QComboBox * m_doubleClickBehavior = nullptr;
QCheckBox * m_useLightCheckerboard = nullptr; QComboBox * m_mouseWheelBehavior = nullptr;
QCheckBox * m_loopGallery = nullptr; QComboBox * m_initWindowSizeBehavior = nullptr;
QComboBox * m_doubleClickBehavior = nullptr; QComboBox * m_hiDpiRoundingPolicyBehavior = nullptr;
QComboBox * m_mouseWheelBehavior = nullptr; };
QComboBox * m_initWindowSizeBehavior = nullptr;
QComboBox * m_hiDpiRoundingPolicyBehavior = nullptr; #endif // SETTINGSDIALOG_H
};
#endif // SETTINGSDIALOG_H

View File

@ -1,126 +0,0 @@
// SPDX-FileCopyrightText: 2024 Gary Wang <opensource@blumia.net>
//
// SPDX-License-Identifier: MIT
#include "shortcutedit.h"
#include <QLabel>
#include <QToolButton>
#include <QHBoxLayout>
#include <QFormLayout>
#include <QDialogButtonBox>
#include <QKeySequenceEdit>
ShortcutEditor::ShortcutEditor(ShortcutEdit * shortcutEdit, QWidget * parent)
: QWidget(parent)
, m_descriptionLabel(new QLabel)
, m_shortcutEdit(shortcutEdit)
, m_shortcutLayout(new QFormLayout)
{
Q_CHECK_PTR(m_shortcutEdit);
QDialogButtonBox * buttons = new QDialogButtonBox(QDialogButtonBox::Apply | QDialogButtonBox::Discard);
QVBoxLayout * layout = new QVBoxLayout(this);
layout->addWidget(m_descriptionLabel);
layout->addLayout(m_shortcutLayout);
layout->addWidget(buttons);
connect(buttons, &QDialogButtonBox::clicked, this, [=](QAbstractButton *button){
if ((QPushButton *)button == buttons->button(QDialogButtonBox::Apply)) {
applyShortcuts();
} else {
reloadShortcuts();
}
});
connect(shortcutEdit, &ShortcutEdit::shortcutsChanged, this, &ShortcutEditor::reloadShortcuts);
setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Minimum);
reloadShortcuts();
}
void ShortcutEditor::setDescription(const QString &desc)
{
m_descriptionLabel->setText(desc);
}
void ShortcutEditor::reloadShortcuts()
{
if (!m_keySequenceEdits.isEmpty()) {
for (QKeySequenceEdit * keyseqEdit : m_keySequenceEdits) {
m_shortcutLayout->removeRow(keyseqEdit);
}
m_keySequenceEdits.clear();
}
QList<QKeySequence> shortcuts = m_shortcutEdit->shortcuts();
shortcuts.append(QKeySequence());
for (const QKeySequence & shortcut : shortcuts) {
QKeySequenceEdit * keyseqEdit = new QKeySequenceEdit(this);
keyseqEdit->setClearButtonEnabled(true);
#if QT_VERSION >= QT_VERSION_CHECK(6, 5, 0)
keyseqEdit->setMaximumSequenceLength(1);
#endif // QT_VERSION >= QT_VERSION_CHECK(6, 5, 0)
keyseqEdit->setKeySequence(shortcut);
m_keySequenceEdits.append(keyseqEdit);
}
for (int i = 0; i < m_keySequenceEdits.count(); i++) {
m_shortcutLayout->addRow(tr("Shortcut #%1").arg(i + 1), m_keySequenceEdits.at(i));
}
}
void ShortcutEditor::applyShortcuts()
{
QList<QKeySequence> shortcuts;
for (const QKeySequenceEdit * keyseqEdit : m_keySequenceEdits) {
if (!keyseqEdit->keySequence().isEmpty() && !shortcuts.contains(keyseqEdit->keySequence())) {
shortcuts.append(keyseqEdit->keySequence());
}
}
emit m_shortcutEdit->applyShortcutsRequested(shortcuts);
}
// ----------------------------------------
ShortcutEdit::ShortcutEdit(QWidget *parent)
: QWidget(parent)
, m_shortcutsLabel(new QLabel(this))
, m_setShortcutButton(new QToolButton(this))
{
m_setShortcutButton->setText("...");
QHBoxLayout * layout = new QHBoxLayout(this);
layout->setContentsMargins(0, 0, 0, 0);
layout->addWidget(m_shortcutsLabel, 1);
layout->addWidget(m_setShortcutButton);
connect(this, &ShortcutEdit::shortcutsChanged, this, [=](){
QStringList shortcutTexts;
for (const QKeySequence & shortcut : std::as_const(m_shortcuts)) {
shortcutTexts.append(shortcut.toString(QKeySequence::NativeText));
}
m_shortcutsLabel->setText(shortcutTexts.isEmpty() ? tr("No shortcuts") : shortcutTexts.join(", "));
m_shortcutsLabel->setDisabled(shortcutTexts.isEmpty());
});
connect(m_setShortcutButton, &QToolButton::clicked, this, &ShortcutEdit::editButtonClicked);
adjustSize();
}
ShortcutEdit::~ShortcutEdit()
{
}
QList<QKeySequence> ShortcutEdit::shortcuts() const
{
return m_shortcuts;
}
void ShortcutEdit::setShortcuts(const QList<QKeySequence> &shortcuts)
{
m_shortcuts = shortcuts;
emit shortcutsChanged();
}

View File

@ -1,55 +0,0 @@
// SPDX-FileCopyrightText: 2024 Gary Wang <opensource@blumia.net>
//
// SPDX-License-Identifier: MIT
#pragma once
#include <QWidget>
#include <QList>
#include <QKeySequence>
class QLabel;
class QFormLayout;
class QToolButton;
class QKeySequenceEdit;
class ShortcutEdit;
class ShortcutEditor : public QWidget
{
Q_OBJECT
public:
explicit ShortcutEditor(ShortcutEdit * shortcutEdit, QWidget * parent = nullptr);
~ShortcutEditor() = default;
void setDescription(const QString & desc);
void reloadShortcuts();
void applyShortcuts();
private:
QLabel * m_descriptionLabel;
ShortcutEdit * m_shortcutEdit;
QFormLayout * m_shortcutLayout;
QList<QKeySequenceEdit *> m_keySequenceEdits;
};
class ShortcutEdit : public QWidget
{
Q_OBJECT
Q_PROPERTY(QList<QKeySequence> shortcuts MEMBER m_shortcuts WRITE setShortcuts NOTIFY shortcutsChanged)
public:
explicit ShortcutEdit(QWidget * parent = nullptr);
~ShortcutEdit();
QList<QKeySequence> shortcuts() const;
void setShortcuts(const QList<QKeySequence> &shortcuts);
signals:
void shortcutsChanged();
void editButtonClicked();
void applyShortcutsRequested(QList<QKeySequence> newShortcuts);
private:
QList<QKeySequence> m_shortcuts;
QLabel * m_shortcutsLabel;
QToolButton * m_setShortcutButton;
};

View File

@ -1,38 +1,38 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#include "toolbutton.h" #include "toolbutton.h"
#include "actionmanager.h" #include "actionmanager.h"
#include "opacityhelper.h" #include "opacityhelper.h"
#include <QPainter> #include <QPainter>
#include <QGraphicsOpacityEffect> #include <QGraphicsOpacityEffect>
#include <QPropertyAnimation> #include <QPropertyAnimation>
ToolButton::ToolButton(bool hoverColor, QWidget *parent) ToolButton::ToolButton(bool hoverColor, QWidget *parent)
: QPushButton(parent) : QPushButton(parent)
, m_opacityHelper(new OpacityHelper(this)) , m_opacityHelper(new OpacityHelper(this))
{ {
setFlat(true); setFlat(true);
QString qss = "QPushButton {" QString qss = "QPushButton {"
"background: transparent;" "background: transparent;"
"}"; "}";
if (hoverColor) { if (hoverColor) {
qss += "QPushButton:hover {" qss += "QPushButton:hover {"
"background: red;" "background: red;"
"}"; "}";
} }
setStyleSheet(qss); setStyleSheet(qss);
} }
void ToolButton::setIconResourcePath(const QString &iconp) void ToolButton::setIconResourcePath(const QString &iconp)
{ {
this->setIcon(ActionManager::loadHidpiIcon(iconp, this->iconSize())); this->setIcon(ActionManager::loadHidpiIcon(iconp, this->iconSize()));
} }
void ToolButton::setOpacity(qreal opacity, bool animated) void ToolButton::setOpacity(qreal opacity, bool animated)
{ {
m_opacityHelper->setOpacity(opacity, animated); m_opacityHelper->setOpacity(opacity, animated);
} }

View File

@ -1,25 +1,25 @@
// SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com> // SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
// //
// SPDX-License-Identifier: MIT // SPDX-License-Identifier: MIT
#ifndef TOOLBUTTON_H #ifndef TOOLBUTTON_H
#define TOOLBUTTON_H #define TOOLBUTTON_H
#include <QPushButton> #include <QPushButton>
class OpacityHelper; class OpacityHelper;
class ToolButton : public QPushButton class ToolButton : public QPushButton
{ {
Q_OBJECT Q_OBJECT
public: public:
ToolButton(bool hoverColor = false, QWidget * parent = nullptr); ToolButton(bool hoverColor = false, QWidget * parent = nullptr);
void setIconResourcePath(const QString &iconp); void setIconResourcePath(const QString &iconp);
public slots: public slots:
void setOpacity(qreal opacity, bool animated = true); void setOpacity(qreal opacity, bool animated = true);
private: private:
OpacityHelper * m_opacityHelper; OpacityHelper * m_opacityHelper;
}; };
#endif // TOOLBUTTON_H #endif // TOOLBUTTON_H

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@ -1,886 +0,0 @@
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE TS>
<TS version="2.1" language="ta">
<context>
<name>AboutDialog</name>
<message>
<location filename="../aboutdialog.cpp" line="29"/>
<source>About</source>
<translation>ி</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="32"/>
<source>Launch application with image file path as argument to load the file.</source>
<translation> .</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="33"/>
<source>Drag and drop image file onto the window is also supported.</source>
<translation> ி ி.</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="34"/>
<source>None of the operations in this application will alter the pictures on disk.</source>
<translation> ி ி .</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="35"/>
<source>Context menu option explanation:</source>
<translation> ி ி ி:</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="42"/>
<source>Make window stay on top of all other windows.</source>
<translation> .</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="45"/>
<source>Avoid close window accidentally. (eg. by double clicking the window)</source>
<translation> ி. (.. )</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="48"/>
<source>Avoid resetting the zoom/rotation/flip state that was applied to the image view when switching between images.</source>
<translation> ி ி /ி/ிி ி ி.</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="55"/>
<source>Version: %1</source>
<translation>ி: %1</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="66"/>
<source>Logo designed by %1</source>
<translation> %1 ி</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="68"/>
<source>Built with Qt %1 (%2)</source>
<translation>ிி %1 ( %2) </translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="69"/>
<source>Source code</source>
<translation> ி</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="79"/>
<source>Contributors</source>
<translation>ி</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="81"/>
<source>List of contributors on GitHub</source>
<translation>ிி ிி ி</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="82"/>
<source>Thanks to all people who contributed to this project.</source>
<translation> ிி ி ி.</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="86"/>
<source>Translators</source>
<translation>ி</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="87"/>
<source>I would like to thank the following people who volunteered to translate this application.</source>
<translation> ி ி ி ி ிி ிி.</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="131"/>
<source>%1 is built on the following free software libraries:</source>
<comment>Free as in freedom</comment>
<translation>%1 ி ி :</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="155"/>
<source>&amp;Special Thanks</source>
<translation>&amp; ி ி</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="157"/>
<source>&amp;Third-party Libraries</source>
<translation> </translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="93"/>
<source>Your Rights</source>
<translation> ி</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="63"/>
<source>Copyright (c) %1 %2</source>
<comment>%1 is year, %2 is the name of copyright holder(s)</comment>
<translation>ிி (ி) %1 %2</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="95"/>
<source>%1 is released under the MIT License.</source>
<translation>%1 ி ிி ிிி.</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="96"/>
<source>This license grants people a number of freedoms:</source>
<translation> ி ி ி:</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="97"/>
<source>You are free to use %1, for any purpose</source>
<translation> ி %1 </translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="98"/>
<source>You are free to distribute %1</source>
<translation>%1 ிிி </translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="99"/>
<source>You can study how %1 works and change it</source>
<translation>%1 ி ி </translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="100"/>
<source>You can distribute changed versions of %1</source>
<translation> ி %1 ிிி</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="102"/>
<source>The MIT license guarantees you this freedom. Nobody is ever permitted to take it away.</source>
<translation>ி ி ி ி. ிி.</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="130"/>
<source>Third-party Libraries used by %1</source>
<translation>%1 </translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="153"/>
<source>&amp;Help</source>
<translation>ி (&amp;h)</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="154"/>
<source>&amp;About</source>
<translation>&amp;ி</translation>
</message>
<message>
<location filename="../aboutdialog.cpp" line="156"/>
<source>&amp;License</source>
<translation>&amp; ி</translation>
</message>
</context>
<context>
<name>GraphicsScene</name>
<message>
<location filename="../mainwindow.cpp" line="276"/>
<location filename="../graphicsscene.cpp" line="100"/>
<source>Drag image here</source>
<translation> </translation>
</message>
</context>
<context>
<name>GraphicsView</name>
<message>
<location filename="../graphicsview.cpp" line="50"/>
<source>File is not a valid image</source>
<translation> ி </translation>
</message>
<message>
<location filename="../graphicsview.cpp" line="54"/>
<location filename="../graphicsview.cpp" line="58"/>
<source>Image data is invalid or currently unsupported</source>
<translation> ி</translation>
</message>
</context>
<context>
<name>MainWindow</name>
<message>
<location filename="../mainwindow.cpp" line="181"/>
<location filename="../mainwindow.cpp" line="545"/>
<source>File url list is empty</source>
<translation> ி ி ி </translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="445"/>
<source>&amp;Copy</source>
<translation> (&amp;c)</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="553"/>
<source>Image data is invalid</source>
<translation> </translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="560"/>
<source>Not supported mimedata: %1</source>
<translation> ிி: %1</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="759"/>
<source>Image From Clipboard</source>
<translation>ிிிி </translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="777"/>
<source>Are you sure you want to move &quot;%1&quot; to recycle bin?</source>
<translation>ி ி &quot;%1&quot; ிி?</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="781"/>
<source>Failed to move file to trash</source>
<translation> ி ி</translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="782"/>
<source>Move to trash failed, it might caused by file permission issue, file system limitation, or platform limitation.</source>
<translation> ி, ி, ி .</translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="114"/>
<source>Copy P&amp;ixmap</source>
<translation>ி &amp; </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="115"/>
<source>Copy &amp;File Path</source>
<translation> </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="133"/>
<source>Properties</source>
<translation></translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="118"/>
<location filename="../aboutdialog.cpp" line="41"/>
<source>Stay on top</source>
<translation> </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="119"/>
<location filename="../aboutdialog.cpp" line="44"/>
<source>Protected mode</source>
<translation> </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="120"/>
<location filename="../aboutdialog.cpp" line="47"/>
<source>Keep transformation</source>
<comment>The &apos;transformation&apos; means the flip/rotation status that currently applied to the image view</comment>
<translation> </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="99"/>
<source>Zoom in</source>
<translation>ி</translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="100"/>
<source>Zoom out</source>
<translation>ிி</translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="108"/>
<source>Pause/Resume Animation</source>
<translation>ி/ி </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="109"/>
<source>Animation Go to Next Frame</source>
<translation>ி ி </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="111"/>
<source>Flip &amp;Horizontally</source>
<translation>ி </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="112"/>
<source>Fit to view</source>
<translation> </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="113"/>
<source>Fit to width</source>
<translation>ி </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="116"/>
<source>&amp;Paste</source>
<translation> (&amp;p)</translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="101"/>
<source>Toggle Checkerboard</source>
<translation> </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="95"/>
<source>&amp;Open...</source>
<translation>&amp; ி ...</translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="97"/>
<source>Actual size</source>
<translation> </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="98"/>
<source>Toggle maximize</source>
<translation>ி </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="102"/>
<source>Rotate right</source>
<translation> </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="103"/>
<source>Rotate left</source>
<translation> </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="105"/>
<source>Previous image</source>
<translation> </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="106"/>
<source>Next image</source>
<translation> </translation>
</message>
<message>
<location filename="../mainwindow.cpp" line="776"/>
<location filename="../actionmanager.cpp" line="117"/>
<source>Move to Trash</source>
<translation> </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="121"/>
<source>Configure...</source>
<translation> ...</translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="122"/>
<source>Help</source>
<translation>ி</translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="125"/>
<source>Show in File Explorer</source>
<comment>File Explorer is the name of explorer.exe under Windows</comment>
<translation> ி ி</translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="131"/>
<source>Show in directory</source>
<translation>ி </translation>
</message>
<message>
<location filename="../actionmanager.cpp" line="134"/>
<source>Quit</source>
<translation>ி</translation>
</message>
</context>
<context>
<name>MetadataDialog</name>
<message>
<location filename="../metadatadialog.cpp" line="84"/>
<source>Image Metadata</source>
<translation> ி </translation>
</message>
</context>
<context>
<name>MetadataModel</name>
<message>
<location filename="../metadatamodel.cpp" line="43"/>
<source>Origin</source>
<comment>Section name.</comment>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="44"/>
<source>Image</source>
<comment>Section name.</comment>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="48"/>
<source>File</source>
<comment>Section name.</comment>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="45"/>
<source>Camera</source>
<comment>Section name.</comment>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="35"/>
<source>%1 File</source>
<translation>%1 </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="42"/>
<source>Description</source>
<comment>Section name.</comment>
<translation>ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="46"/>
<source>Advanced photo</source>
<comment>Section name.</comment>
<translation> </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="47"/>
<source>GPS</source>
<comment>Section name.</comment>
<translation> </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="52"/>
<source>Dimensions</source>
<translation>ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="54"/>
<source>Aspect ratio</source>
<translation> ிி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="58"/>
<source>Frame count</source>
<translation> ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="62"/>
<source>Name</source>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="64"/>
<source>Item type</source>
<translation>ி </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="66"/>
<source>Folder path</source>
<translation> </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="68"/>
<source>Size</source>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="70"/>
<source>Date created</source>
<translation>ி </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="72"/>
<source>Date modified</source>
<translation>ி ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="79"/>
<source>Title</source>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="81"/>
<source>Subject</source>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="83"/>
<source>Rating</source>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="85"/>
<source>Tags</source>
<translation>ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="87"/>
<source>Comments</source>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="90"/>
<source>Authors</source>
<translation>ிி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="92"/>
<source>Date taken</source>
<translation> ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="98"/>
<source>Program name</source>
<translation>ி </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="100"/>
<source>Copyright</source>
<translation>ிி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="103"/>
<source>Horizontal resolution</source>
<translation>ி </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="105"/>
<source>Vertical resolution</source>
<translation> </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="107"/>
<source>Resolution unit</source>
<translation>ிி </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="109"/>
<source>Colour representation</source>
<translation> ிிிி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="112"/>
<source>Camera maker</source>
<translation> ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="114"/>
<source>Camera model</source>
<translation> ிி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="116"/>
<source>F-stop</source>
<translation>-</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="118"/>
<source>Exposure time</source>
<translation>ி </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="120"/>
<source>ISO speed</source>
<translation> ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="122"/>
<source>Exposure bias</source>
<translation>ி </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="124"/>
<source>Focal length</source>
<translation>ி, ி </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="126"/>
<source>Max aperture</source>
<translation>ி </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="128"/>
<source>Metering mode</source>
<translation> </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="130"/>
<source>Subject distance</source>
<translation> </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="132"/>
<source>Flash mode</source>
<translation>ி </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="134"/>
<source>35mm focal length</source>
<translation>35 ி ி </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="137"/>
<source>Lens model</source>
<translation> ிி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="139"/>
<source>Contrast</source>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="141"/>
<source>Brightness</source>
<translation>ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="143"/>
<source>Exposure program</source>
<translation>ி ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="145"/>
<source>Saturation</source>
<translation>ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="147"/>
<source>Sharpness</source>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="149"/>
<source>White balance</source>
<translation> </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="151"/>
<source>Digital zoom</source>
<translation>ிி </translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="153"/>
<source>EXIF version</source>
<translation>Exif ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="156"/>
<source>Latitude reference</source>
<translation> ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="158"/>
<source>Latitude</source>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="160"/>
<source>Longitude reference</source>
<translation> ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="162"/>
<source>Longitude</source>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="164"/>
<source>Altitude reference</source>
<translation> ி</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="166"/>
<source>Altitude</source>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="176"/>
<source>%1 x %2</source>
<translation>%1 %2</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="194"/>
<source>%1 : %2</source>
<translation>%1: %2</translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="319"/>
<source>Property</source>
<translation></translation>
</message>
<message>
<location filename="../metadatamodel.cpp" line="319"/>
<source>Value</source>
<translation>ி</translation>
</message>
</context>
<context>
<name>SettingsDialog</name>
<message>
<location filename="../settingsdialog.cpp" line="31"/>
<source>Settings</source>
<translation></translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="39"/>
<source>Options</source>
<translation>ி</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="51"/>
<source>Shortcuts</source>
<translation>ி</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="61"/>
<source>Editing shortcuts for action &quot;%1&quot;:</source>
<translation> ி ி &quot;%1&quot;:</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="70"/>
<source>Failed to set shortcuts</source>
<translation>ி ி ி</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="71"/>
<source>Please check if shortcuts are duplicated with existing shortcuts.</source>
<translation> ி ி ி.</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="78"/>
<source>Do nothing</source>
<translation> </translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="79"/>
<source>Close the window</source>
<translation> </translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="80"/>
<source>Toggle maximize</source>
<translation>ி </translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="81"/>
<source>Toggle fullscreen</source>
<translation> ி</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="85"/>
<source>Zoom in and out</source>
<translation> ி ி</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="86"/>
<source>View next or previous item</source>
<translation> ி </translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="90"/>
<source>Auto size</source>
<translation> </translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="91"/>
<source>Maximized</source>
<translation>ி</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="92"/>
<source>Windowed</source>
<translation></translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="96"/>
<source>Round (Integer scaling)</source>
<comment>This option means round up for .5 and above</comment>
<translation> ( ி)</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="97"/>
<source>Ceil (Integer scaling)</source>
<comment>This option means always round up</comment>
<translation> ( ி)</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="98"/>
<source>Floor (Integer scaling)</source>
<comment>This option means always round down</comment>
<translation>ி ( ி)</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="99"/>
<source>Follow system (Fractional scaling)</source>
<comment>This option means don&apos;t round</comment>
<translation>ிி ி (ி ி)</translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="122"/>
<source>Stay on top when start-up</source>
<translation>ி </translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="123"/>
<source>Use built-in close window animation</source>
<translation> ி </translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="124"/>
<source>Use light-color checkerboard</source>
<translation>ி- </translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="125"/>
<source>Loop the loaded gallery</source>
<translation> ி </translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="126"/>
<source>Double-click behavior</source>
<translation> </translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="127"/>
<source>Mouse wheel behavior</source>
<translation>ி </translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="128"/>
<source>Default window size</source>
<translation>ி </translation>
</message>
<message>
<location filename="../settingsdialog.cpp" line="129"/>
<source>HiDPI scale factor rounding policy</source>
<translation>HIDPI ி ி ி </translation>
</message>
</context>
<context>
<name>ShortcutEdit</name>
<message>
<location filename="../shortcutedit.cpp" line="109"/>
<source>No shortcuts</source>
<translation>ி </translation>
</message>
</context>
<context>
<name>ShortcutEditor</name>
<message>
<location filename="../shortcutedit.cpp" line="75"/>
<source>Shortcut #%1</source>
<translation>ி #%1</translation>
</message>
</context>
<context>
<name>main</name>
<message>
<location filename="../main.cpp" line="42"/>
<source>Pineapple Pictures</source>
<translation>ி </translation>
</message>
<message>
<location filename="../main.cpp" line="45"/>
<source>List supported image format suffixes, and quit program.</source>
<translation>ி ி ி ிி, ி ி.</translation>
</message>
<message>
<location filename="../main.cpp" line="49"/>
<source>File list.</source>
<translation> ி.</translation>
</message>
</context>
</TS>

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@ -8,14 +8,22 @@ environment:
LIBEXIV2: C:\projects\exiv2 LIBEXIV2: C:\projects\exiv2
PPKG: C:\projects\ppkg PPKG: C:\projects\ppkg
matrix: matrix:
- job_name: mingw_64_qt6_8 - job_name: mingw_64_qt6_5
QTDIR: C:\Qt\6.8\mingw_64 QTDIR: C:\Qt\6.5\mingw_64
MINGW64: C:\Qt\Tools\mingw1310_64 MINGW64: C:\Qt\Tools\mingw1120_64
KF_BRANCH: master KF_BRANCH: master
EXIV2_VERSION: "0.28.5" EXIV2_VERSION: "0.28.0"
EXIV2_CMAKE_OPTIONS: "-DEXIV2_ENABLE_BROTLI=OFF -DEXIV2_ENABLE_INIH=OFF -DEXIV2_BUILD_EXIV2_COMMAND=OFF" EXIV2_CMAKE_OPTIONS: "-DEXIV2_ENABLE_BROTLI=OFF -DEXIV2_ENABLE_INIH=OFF -DEXIV2_BUILD_EXIV2_COMMAND=OFF"
PPIC_CMAKE_OPTIONS: "-DPREFER_QT_5=OFF" PPIC_CMAKE_OPTIONS: "-DPREFER_QT_5=OFF"
WINDEPLOYQT_ARGS: "--verbose=2 --no-quick-import --no-translations --no-opengl-sw --no-system-d3d-compiler --skip-plugin-types tls,networkinformation" 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: install:
- mkdir %CMAKE_INSTALL_PREFIX% - mkdir %CMAKE_INSTALL_PREFIX%
@ -33,21 +41,36 @@ install:
build_script: build_script:
# prepare # prepare
- mkdir 3rdparty - mkdir 3rdparty
- choco install ninja - cinst ninja
- cd %PPKG% - cd %PPKG%
- curl -fsSL -o ppkg.exe https://github.com/BLumia/pineapple-package-manager/releases/latest/download/ppkg.exe - curl -fsSL -o ppkg.exe https://github.com/BLumia/pineapple-package-manager/releases/latest/download/ppkg.exe
- cd %APPVEYOR_BUILD_FOLDER% - cd %APPVEYOR_BUILD_FOLDER%
# download and install zlib for KArchive # download and install zlib for KArchive
- cd %LIBZ% - cd %LIBZ%
- curl -fsSL -o zlib131.zip https://zlib.net/zlib131.zip - curl -fsSL -o zlib13.zip https://zlib.net/zlib13.zip
- 7z x zlib131.zip -y - 7z x zlib13.zip -y
- cd zlib-1.3.1 - cd zlib-1.3
- mkdir build - mkdir build
- cd build - cd build
- cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_PREFIX% - cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_PREFIX%
- cmake --build . --config Release - cmake --build . --config Release
- cmake --build . --config Release --target install/strip - cmake --build . --config Release --target install/strip
- cd %APPVEYOR_BUILD_FOLDER% - cd %APPVEYOR_BUILD_FOLDER%
# QtApng for APNG
- cd 3rdparty
- mkdir QtApng
- cd QtApng
- curl -fsSL -o ppkg-QtApng.zip https://sourceforge.net/projects/pineapple-package-manager/files/packages/mingw-w64-x86_64-windows/QtApng-qt5.15.2-gea53d73-1.zip
- ppkg ppkg-QtApng.zip
- 7z x ppkg-QtApng.zip LICENSE -y
# - git clone --branch detect-format-from-content --depth 1 https://github.com/BLumia/QtApng.git
# - cd QtApng
# - mkdir build
# - cd build
# - qmake "CONFIG+=libpng_static" ../
# - mingw32-make
# - mingw32-make install
- cd %APPVEYOR_BUILD_FOLDER%
# install ECM so we can build KImageFormats # install ECM so we can build KImageFormats
- cd 3rdparty - cd 3rdparty
- git clone -b %KF_BRANCH% -q https://invent.kde.org/frameworks/extra-cmake-modules.git - git clone -b %KF_BRANCH% -q https://invent.kde.org/frameworks/extra-cmake-modules.git
@ -57,9 +80,24 @@ build_script:
- cmake --build . - cmake --build .
- cmake --build . --target install - cmake --build . --target install
- cd %APPVEYOR_BUILD_FOLDER% - cd %APPVEYOR_BUILD_FOLDER%
# install openexr for EXR format plugin from KImageFormats
- cd 3rdparty
- mkdir openexr
- cd openexr
# - git clone -b v3.1.3 --depth 1 https://github.com/AcademySoftwareFoundation/openexr.git
# - cd openexr
# - mkdir build
# - cd build
# - cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DCMAKE_CXX_FLAGS=-D__STDC_FORMAT_MACROS=1 -DOPENEXR_INSTALL_EXAMPLES=OFF -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_PREFIX%
# - cmake --build . --config Release
# - cmake --build . --config Release --target install/strip
- curl -fsSL -o ppkg-openexr.zip https://sourceforge.net/projects/pineapple-package-manager/files/packages/mingw-w64-x86_64-windows/openexr-3.1.3-1.zip
- ppkg ppkg-openexr.zip
- 7z x ppkg-openexr.zip LICENSE.md -y
- cd %APPVEYOR_BUILD_FOLDER%
# install AOM for libavif AV1 decoding support... # install AOM for libavif AV1 decoding support...
- cd 3rdparty - cd 3rdparty
#- git clone -b v3.9.1 --depth 1 https://aomedia.googlesource.com/aom #- git clone -b v3.6.0 --depth 1 https://aomedia.googlesource.com/aom
#- cd aom #- cd aom
#- mkdir build.aom #- mkdir build.aom
#- cd build.aom #- cd build.aom
@ -68,18 +106,18 @@ build_script:
#- cmake --build . --config Release --target install/strip #- cmake --build . --config Release --target install/strip
- mkdir aom - mkdir aom
- cd 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.9.1-2.zip - 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 - ppkg ppkg-aom.zip
- 7z x ppkg-aom.zip LICENSE -y - 7z x ppkg-aom.zip LICENSE -y
- cd %APPVEYOR_BUILD_FOLDER% - cd %APPVEYOR_BUILD_FOLDER%
# install libavif for avif format support of KImageFormats # install libavif for avif format support of KImageFormats
- cd %LIBAVIF% - cd %LIBAVIF%
- curl -fsSL -o libavif-v1_1_1.zip https://github.com/AOMediaCodec/libavif/archive/v1.1.1.zip - curl -fsSL -o libavif-v0_11_1.zip https://github.com/AOMediaCodec/libavif/archive/v0.11.1.zip
- 7z x libavif-v1_1_1.zip -y - 7z x libavif-v0_11_1.zip -y
- cd libavif-1.1.1 - cd libavif-0.11.1
- mkdir build - mkdir build
- cd build - cd build
- cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_PREFIX% -DAVIF_CODEC_AOM=ON -DAVIF_LOCAL_LIBYUV=ON - 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
- cmake --build . --config Release --target install/strip - cmake --build . --config Release --target install/strip
- cd %APPVEYOR_BUILD_FOLDER% - cd %APPVEYOR_BUILD_FOLDER%
@ -90,15 +128,15 @@ build_script:
- cd karchive - cd karchive
- mkdir build - mkdir build
- cd build - cd build
- cmake .. -G "Ninja" -DBUILD_TESTING=OFF -DWITH_LIBZSTD=OFF -DWITH_BZIP2=OFF -DWITH_LIBLZMA=OFF -DWITH_OPENSSL=OFF -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_PREFIX% - cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=%CMAKE_INSTALL_PREFIX%
- cmake --build . --config Release - cmake --build . --config Release
- cmake --build . --config Release --target install/strip - cmake --build . --config Release --target install/strip
- cd %APPVEYOR_BUILD_FOLDER% - cd %APPVEYOR_BUILD_FOLDER%
# build libexpat for libexiv2 # build libexpat for libexiv2
- cd %LIBEXPAT% - cd %LIBEXPAT%
- curl -fsSL -o R_2_6_2.zip https://github.com/libexpat/libexpat/archive/R_2_6_2.zip - curl -fsSL -o R_2_5_0.zip https://github.com/libexpat/libexpat/archive/R_2_5_0.zip
- 7z x R_2_6_2.zip -y - 7z x R_2_5_0.zip -y
- cd libexpat-R_2_6_2/expat/ - 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 -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 - cmake --build . --target install/strip
- cd %APPVEYOR_BUILD_FOLDER% - cd %APPVEYOR_BUILD_FOLDER%
@ -117,14 +155,14 @@ build_script:
- cd kimageformats - cd kimageformats
- mkdir build - mkdir build
- cd build - cd build
- cmake .. -G "Ninja" -DBUILD_TESTING=OFF -DCMAKE_BUILD_TYPE=Release -DKDE_INSTALL_QTPLUGINDIR=%QTDIR%\plugins - cmake .. -G "Ninja" -DCMAKE_BUILD_TYPE=Release -DKDE_INSTALL_QTPLUGINDIR=%QTDIR%\plugins
- cmake --build . --config Release - cmake --build . --config Release
- cmake --build . --config Release --target install/strip - cmake --build . --config Release --target install/strip
- cd %APPVEYOR_BUILD_FOLDER% - cd %APPVEYOR_BUILD_FOLDER%
# finally... # finally...
- mkdir build - mkdir build
- cd build - cd build
- cmake .. -G "Ninja" %PPIC_CMAKE_OPTIONS% -DCMAKE_BUILD_TYPE=Release -DCMAKE_PREFIX_PATH=%CMAKE_INSTALL_PREFIX% -DCMAKE_INSTALL_PREFIX='%cd%' - 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
- cmake --build . --config Release --target install/strip - cmake --build . --config Release --target install/strip
# fixme: I don't know how to NOT make the binary installed to the ./bin/ folder... # fixme: I don't know how to NOT make the binary installed to the ./bin/ folder...
@ -135,16 +173,22 @@ build_script:
- copy %CMAKE_INSTALL_PREFIX%\bin\libexiv2.dll - copy %CMAKE_INSTALL_PREFIX%\bin\libexiv2.dll
- copy %CMAKE_INSTALL_PREFIX%\bin\libavif.dll - copy %CMAKE_INSTALL_PREFIX%\bin\libavif.dll
- copy %CMAKE_INSTALL_PREFIX%\bin\libzlib.dll - copy %CMAKE_INSTALL_PREFIX%\bin\libzlib.dll
- copy %CMAKE_INSTALL_PREFIX%\bin\libIex-3_1.dll
- copy %CMAKE_INSTALL_PREFIX%\bin\libImath-3_1.dll
- copy %CMAKE_INSTALL_PREFIX%\bin\libIlmThread-3_1.dll
- copy %CMAKE_INSTALL_PREFIX%\bin\libOpenEXR-3_1.dll
- copy %CMAKE_INSTALL_PREFIX%\bin\libKF?Archive.dll - copy %CMAKE_INSTALL_PREFIX%\bin\libKF?Archive.dll
- windeployqt %WINDEPLOYQT_ARGS% .\ppic.exe - windeployqt %WINDEPLOYQT_ARGS% .\ppic.exe
# copy 3rdparty licenses for the libs we vendored for windows... # copy 3rdparty licenses for the libs we vendored for windows...
- mkdir licenses - mkdir licenses
- cd licenses - cd licenses
- copy %APPVEYOR_BUILD_FOLDER%\3rdparty\aom\LICENSE License.aom.txt - copy %APPVEYOR_BUILD_FOLDER%\3rdparty\aom\LICENSE License.aom.txt
- copy %APPVEYOR_BUILD_FOLDER%\3rdparty\openexr\LICENSE.md License.openexr.txt
- copy %APPVEYOR_BUILD_FOLDER%\3rdparty\karchive\LICENSES\LGPL-2.0-or-later.txt License.KArchive.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 %APPVEYOR_BUILD_FOLDER%\3rdparty\kimageformats\LICENSES\LGPL-2.1-or-later.txt License.kimageformats.txt
- copy %LIBEXPAT%\libexpat-R_2_6_2\expat\COPYING License.expat.txt - copy %APPVEYOR_BUILD_FOLDER%\3rdparty\QtApng\LICENSE License.QtApng.txt
- copy %LIBAVIF%\libavif-1.1.1\LICENSE License.libavif.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 - copy %LIBEXIV2%\exiv2-%EXIV2_VERSION%\COPYING License.exiv2.txt
# TODO: Qt, zlib # TODO: Qt, zlib
- cd .. - cd ..

Binary file not shown.

View File

@ -6,7 +6,7 @@ BEGIN
BLOCK "040904E4" BLOCK "040904E4"
BEGIN BEGIN
VALUE "FileDescription", "Pineapple Pictures - Image Viewer" VALUE "FileDescription", "Pineapple Pictures - Image Viewer"
VALUE "LegalCopyright", "MIT/Expat License - Copyright (C) 2024 Gary Wang" VALUE "LegalCopyright", "MIT/Expat License - Copyright (C) 2020 Gary Wang"
VALUE "ProductName", "Pineapple Pictures" VALUE "ProductName", "Pineapple Pictures"
END END
END END
@ -14,4 +14,4 @@ BEGIN
BEGIN BEGIN
VALUE "Translation", 0x409, 1200 VALUE "Translation", 0x409, 1200
END END
END END

View File

@ -4,16 +4,15 @@
<li><u>Dutch</u>: Heimen Stoffels</li> <li><u>Dutch</u>: Heimen Stoffels</li>
<li><u>French</u>: J. Lavoie, K. Herbert, Maxime Leroy</li> <li><u>French</u>: J. Lavoie, K. Herbert, Maxime Leroy</li>
<li><u>German</u>: K. Herbert, J. Lavoie, sal0max</li> <li><u>German</u>: K. Herbert, J. Lavoie, sal0max</li>
<li><u>Indonesian</u>: liimee, Reza Almanda</li> <li><u>Indonesian</u>: liimee</li>
<li><u>Italian</u>: albanobattistella</li> <li><u>Italian</u>: albanobattistella</li>
<li><u>Japanese</u>: Black Cat, mmahhi, Percy Hong</li> <li><u>Japanese</u>: Black Cat, Percy Hong</li>
<li><u>Korean</u>: VenusGirl</li> <li><u>Korean</u>: VenusGirl</li>
<li><u>Norwegian Bokmål</u>: Allan Nordhøy, ovl-1</li> <li><u>Norwegian Bokmål</u>: Allan Nordhøy, ovl-1</li>
<li><u>Punjabi (Pakistan)</u>: bgo-eiu</li> <li><u>Punjabi (Pakistan)</u>: bgo-eiu</li>
<li><u>Russian</u>: Sergey Shornikov, Artem, Andrey</li> <li><u>Russian</u>: Sergey Shornikov, Artem, Andrey</li>
<li><u>Sinhala</u>: HelaBasa</li> <li><u>Sinhala</u>: HelaBasa</li>
<li><u>Spanish</u>: Toni Estévez, Génesis Toxical, William(ѕ)ⁿ, gallegonovato</li> <li><u>Spanish</u>: Toni Estévez, Génesis Toxical, William(ѕ)ⁿ, gallegonovato</li>
<li><u>Tamil</u>: தமிழ்நேரம் (TamilNeram)</li>
<li><u>Turkish</u>: E-Akcaer, Oğuz Ersen, Sabri Ünal</li> <li><u>Turkish</u>: E-Akcaer, Oğuz Ersen, Sabri Ünal</li>
<li><u>Ukrainian</u>: Dan, volkov, Сергій</li> <li><u>Ukrainian</u>: Dan</li>
</ul> </ul>

96
cmake/FindLibExiv2.cmake Normal file
View File

@ -0,0 +1,96 @@
#.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 <christophe@krop.fr>
# SPDX-FileCopyrightText: 2010 Alexander Neundorf <neundorf@kde.org>
# SPDX-FileCopyrightText: 2008 Gilles Caulier <caulier.gilles@gmail.com>
#
# 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"
)

View File

@ -1,136 +0,0 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple Computer//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>CFBundleDevelopmentRegion</key>
<string>English</string>
<key>CFBundleExecutable</key>
<string>${MACOSX_BUNDLE_EXECUTABLE_NAME}</string>
<key>CFBundleGetInfoString</key>
<string>${MACOSX_BUNDLE_INFO_STRING}</string>
<key>CFBundleIconFile</key>
<string>${MACOSX_BUNDLE_ICON_FILE}</string>
<key>CFBundleIdentifier</key>
<string>${MACOSX_BUNDLE_GUI_IDENTIFIER}</string>
<key>CFBundleInfoDictionaryVersion</key>
<string>6.0</string>
<key>CFBundleLongVersionString</key>
<string>${MACOSX_BUNDLE_LONG_VERSION_STRING}</string>
<key>CFBundleName</key>
<string>${MACOSX_BUNDLE_BUNDLE_NAME}</string>
<key>CFBundlePackageType</key>
<string>APPL</string>
<key>CFBundleShortVersionString</key>
<string>${MACOSX_BUNDLE_SHORT_VERSION_STRING}</string>
<key>CFBundleSignature</key>
<string>????</string>
<key>CFBundleVersion</key>
<string>${MACOSX_BUNDLE_BUNDLE_VERSION}</string>
<key>CSResourcesFileMapped</key>
<true/>
<key>NSHumanReadableCopyright</key>
<string>${MACOSX_BUNDLE_COPYRIGHT}</string>
<!-- FIXME: this list can't be automatically generated by Qt's CMake API, don't know why. -->
<key>CFBundleLocalizations</key>
<array>
<string>en</string>
<string>ca</string>
<string>de</string>
<string>es</string>
<string>fr</string>
<string>id</string>
<string>it</string>
<string>ja</string>
<string>ko</string>
<string>nb_NO</string>
<string>nl</string>
<string>pa_PK</string>
<string>ru</string>
<string>si</string>
<string>ta</string>
<string>tr</string>
<string>uk</string>
<string>zh_CN</string>
</array>
<key>CFBundleDocumentTypes</key>
<array>
<!-- JPEG -->
<dict>
<key>CFBundleTypeName</key>
<string>JPEG Image</string>
<key>CFBundleTypeExtensions</key>
<array>
<string>jpg</string>
<string>jpeg</string>
<string>jfif</string>
</array>
<key>CFBundleTypeMIMETypes</key>
<array>
<string>image/jpeg</string>
</array>
<key>CFBundleTypeRole</key>
<string>Viewer</string>
</dict>
<!-- PNG -->
<dict>
<key>CFBundleTypeName</key>
<string>PNG Image</string>
<key>CFBundleTypeExtensions</key>
<array>
<string>png</string>
</array>
<key>CFBundleTypeMIMETypes</key>
<array>
<string>image/png</string>
</array>
<key>CFBundleTypeRole</key>
<string>Viewer</string>
</dict>
<!-- WebP -->
<dict>
<key>CFBundleTypeName</key>
<string>WebP Image</string>
<key>CFBundleTypeExtensions</key>
<array>
<string>webp</string>
</array>
<key>CFBundleTypeMIMETypes</key>
<array>
<string>image/webp</string>
</array>
<key>CFBundleTypeRole</key>
<string>Viewer</string>
</dict>
<!-- GIF -->
<dict>
<key>CFBundleTypeName</key>
<string>GIF Image</string>
<key>CFBundleTypeExtensions</key>
<array>
<string>gif</string>
</array>
<key>CFBundleTypeMIMETypes</key>
<array>
<string>image/gif</string>
</array>
<key>CFBundleTypeRole</key>
<string>Viewer</string>
</dict>
<!-- SVG -->
<dict>
<key>CFBundleTypeName</key>
<string>SVG Image</string>
<key>CFBundleTypeExtensions</key>
<array>
<string>svg</string>
</array>
<key>CFBundleTypeMIMETypes</key>
<array>
<string>image/svg+xml</string>
</array>
<key>CFBundleTypeRole</key>
<string>Viewer</string>
</dict>
</array>
</dict>
</plist>

View File

@ -1,264 +1,59 @@
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<component type="desktop-application"> <component type="desktop-application">
<id>net.blumia.pineapple-pictures</id> <id>net.blumia.pineapple-pictures</id>
<name>Pineapple Pictures</name> <launchable type="desktop-id">net.blumia.pineapple-pictures.desktop</launchable>
<name xml:lang="es">Pineapple Pictures</name>
<name xml:lang="ja">Pineapple Pictures</name>
<name xml:lang="nl">Pineapple Afbeeldingen</name>
<name xml:lang="ru">Pineapple Pictures</name>
<name xml:lang="ta">அன்னாசி படங்கள்</name>
<name xml:lang="uk">Pineapple Pictures</name>
<name xml:lang="zh-CN">菠萝看图</name>
<summary>Image Viewer</summary>
<summary xml:lang="es">Visor de imágenes</summary>
<summary xml:lang="ja">画像ビューアー</summary>
<summary xml:lang="nl">Afbeeldingsweergave</summary>
<summary xml:lang="ru">Просмотр изображений</summary>
<summary xml:lang="ta">பட பார்வையாளர்</summary>
<summary xml:lang="uk">Переглядач зображень</summary>
<summary xml:lang="zh-CN">图像查看器</summary>
<metadata_license>CC0-1.0</metadata_license> <metadata_license>CC0-1.0</metadata_license>
<project_license>MIT</project_license> <project_license>MIT</project_license>
<name>Pineapple Pictures</name>
<name xml:lang="nl">Pineapple Afbeeldingen</name>
<name xml:lang="ru">Pineapple Pictures</name>
<name xml:lang="zh-CN">菠萝看图</name>
<summary>Image Viewer</summary>
<summary xml:lang="nl">Afbeeldingsweergave</summary>
<summary xml:lang="ru">Просмотр изображений</summary>
<summary xml:lang="zh-CN">图像查看器</summary>
<description> <description>
<p>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.</p> <p>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.</p>
<p xml:lang="es">Pineapple Pictures es un visor de imágenes ligero y fácil de usar que viene con una práctica miniatura de navegación al hacer zoom, y no contiene ningún soporte de gestión de imágenes.</p>
<p xml:lang="ja">Pineapple Picturesは、ズームイン時に便利なナビゲーションサムネイルを備えた軽量で使いやすい画像ビューアです。画像管理のサポートは含まれていません。</p>
<p xml:lang="nl">Pineapple Afbeeldingen is een licht en eenvoudig te gebruiken afbeeldingsweergaveprogramma met miniatuurnavigatie na inzoomen. Het programma heeft echter geen fotobeheermogelijkheid.</p> <p xml:lang="nl">Pineapple Afbeeldingen is een licht en eenvoudig te gebruiken afbeeldingsweergaveprogramma met miniatuurnavigatie na inzoomen. Het programma heeft echter geen fotobeheermogelijkheid.</p>
<p xml:lang="ru">Pineapple Pictures - это легкий и простой в использовании просмотрщик изображений, оснащенный удобной навигацией по миниатюрам при увеличении масштаба и не содержащий никакой поддержки управления изображениями.</p> <p xml:lang="ru">Pineapple Pictures - это легкий и простой в использовании просмотрщик изображений, оснащенный удобной навигацией по миниатюрам при увеличении масштаба и не содержащий никакой поддержки управления изображениями.</p>
<p xml:lang="ta">அன்னாசி படங்கள் ஒரு இலகுரக மற்றும் பயன்படுத்த எளிதான பட பார்வையாளராகும், இது பெரிதாக்கும்போது ஒரு எளிமையான வழிசெலுத்தல் சிறுபடத்துடன் வருகிறது, மேலும் எந்த பட மேலாண்மை ஆதரவையும் கொண்டிருக்கவில்லை.</p>
<p xml:lang="uk">Pineapple Pictures це легкий і простий у використанні переглядач зображень, який постачається зі зручною навігаційною мініатюрою при збільшенні масштабу і не містить жодної підтримки керування зображеннями.</p>
<p xml:lang="zh-CN">菠萝看图是一个轻量级易用的图像查看器,在图片放大时提供了方便的鸟瞰导航功能,且不包含任何图片管理功能。</p> <p xml:lang="zh-CN">菠萝看图是一个轻量级易用的图像查看器,在图片放大时提供了方便的鸟瞰导航功能,且不包含任何图片管理功能。</p>
</description> </description>
<developer id="net.blumia">
<name>Gary (BLumia) Wang et al.</name>
<name xml:lang="ja">Gary (BLumia) Wang など</name>
<name xml:lang="nl">Gary (BLumia) Wang e.a.</name>
<name xml:lang="ru">Gary (BLumia) Wang et al.</name>
<name xml:lang="uk">Gary (BLumia) Wang.</name>
<name xml:lang="zh-CN">Gary (BLumia) Wang 等人</name>
</developer>
<launchable type="desktop-id">net.blumia.pineapple-pictures.desktop</launchable>
<url type="homepage">https://github.com/BLumia/pineapple-pictures</url>
<url type="bugtracker">https://github.com/BLumia/pineapple-pictures/issues</url>
<url type="translate">https://hosted.weblate.org/projects/pineapple-pictures/</url>
<provides>
<binary>ppic</binary>
</provides>
<screenshots> <screenshots>
<screenshot type="default"> <screenshot type="default">
<caption>Main window when an image file is loaded</caption> <caption>Main window when an image file is loaded</caption>
<caption xml:lang="es">Ventana principal cuando se carga un archivo de imagen</caption>
<caption xml:lang="ja">画像ファイル読み込み時のメインウィンドウ</caption>
<caption xml:lang="nl">Hoofdvenster na het laden van een afbeelding</caption> <caption xml:lang="nl">Hoofdvenster na het laden van een afbeelding</caption>
<caption xml:lang="ru">Основное окно после загрузки файла изображения</caption> <caption xml:lang="ru">Основное окно после загрузки файла изображения</caption>
<caption xml:lang="ta">ஒரு படக் கோப்பு ஏற்றப்படும் போது முதன்மையான சாளரம்</caption>
<caption xml:lang="uk">Головне вікно після завантаження файлу зображення</caption>
<caption xml:lang="zh-CN">加载图片后的主窗口</caption> <caption xml:lang="zh-CN">加载图片后的主窗口</caption>
<image type="source" width="1503" height="640">https://pineapple-pictures.sourceforge.io/ppic-gui-static.png</image> <image type="source" width="1503" height="640">https://pineapple-pictures.sourceforge.io/ppic-gui-static.png</image>
</screenshot> </screenshot>
<screenshot> <screenshot>
<caption>Zooming in a raster image</caption> <caption>Zooming in a raster image</caption>
<caption xml:lang="es">Ampliar una imagen rasterizada</caption>
<caption xml:lang="ja">ラスター画像の拡大</caption>
<caption xml:lang="nl">Inzoomen op een roosterafbeelding</caption> <caption xml:lang="nl">Inzoomen op een roosterafbeelding</caption>
<caption xml:lang="ru">Масштабирование растрового изображения</caption> <caption xml:lang="ru">Масштабирование растрового изображения</caption>
<caption xml:lang="ta">ராச்டர் படத்தில் பெரிதாக்குதல்</caption>
<caption xml:lang="uk">Масштабування растрового зображення</caption>
<caption xml:lang="zh-CN">放大浏览位图</caption> <caption xml:lang="zh-CN">放大浏览位图</caption>
<image type="source" width="771" height="553">https://pineapple-pictures.sourceforge.io/ppic-zoom-raster.png</image> <image type="source" width="771" height="553">https://pineapple-pictures.sourceforge.io/ppic-zoom-raster.png</image>
</screenshot> </screenshot>
<screenshot> <screenshot>
<caption>Zooming in a vector image</caption> <caption>Zooming in a vector image</caption>
<caption xml:lang="es">Ampliar una imagen vectorial</caption>
<caption xml:lang="ja">ベクター画像の拡大</caption>
<caption xml:lang="nl">Inzoomen op een vectorafbeelding</caption> <caption xml:lang="nl">Inzoomen op een vectorafbeelding</caption>
<caption xml:lang="ru">Масштабирование векторного изображения</caption> <caption xml:lang="ru">Масштабирование векторного изображения</caption>
<caption xml:lang="ta">ஒரு திசையன் படத்தில் பெரிதாக்குதல்</caption>
<caption xml:lang="uk">Масштабування векторного зображення</caption>
<caption xml:lang="zh-CN">放大浏览矢量图</caption> <caption xml:lang="zh-CN">放大浏览矢量图</caption>
<image type="source" width="771" height="553">https://pineapple-pictures.sourceforge.io/ppic-zoom-svg.png</image> <image type="source" width="771" height="553">https://pineapple-pictures.sourceforge.io/ppic-zoom-svg.png</image>
</screenshot> </screenshot>
</screenshots> </screenshots>
<releases> <url type="homepage">https://github.com/BLumia/pineapple-pictures</url>
<release type="stable" version="1.1.0" date="2025-07-06T00:00:00Z"> <url type="bugtracker">https://github.com/BLumia/pineapple-pictures/issues</url>
<description> <url type="translate">https://hosted.weblate.org/projects/pineapple-pictures/</url>
<p>This release adds the following features:</p> <developer_name>Gary (BLumia) Wang et al.</developer_name>
<ul> <developer_name xml:lang="nl">Gary (BLumia) Wang e.a.</developer_name>
<li>New option to disable built-in close window animation</li> <developer_name xml:lang="ru">Gary (BLumia) Wang et al.</developer_name>
<li>New option to disable gallery looping</li> <developer_name xml:lang="zh-CN">Gary (BLumia) Wang 等人</developer_name>
<li>Support load m3u8 as image gallery playlist</li> <provides>
</ul> <binary>ppic</binary>
<p>This release includes the following change:</p> </provides>
<ul>
<li>Drop Qt 5 support</li>
</ul>
<p>With contributions from:</p>
<p>Heimen Stoffels, albanobattistella, தமிழ்நேரம்</p>
</description>
</release>
<release type="stable" version="1.0.0" date="2025-05-03T00:00:00Z">
<description>
<p>This release adds the following features:</p>
<ul>
<li>Support enforces windowed mode on start-up</li>
<li>Reload image automatically when current image gets updated</li>
</ul>
<p>This release fixes the following bug:</p>
<ul>
<li>Display correct text language on macOS</li>
</ul>
<p>This release includes the following changes:</p>
<ul>
<li>Use native text for shortcut editor's label</li>
<li>Display native commandline message when possible</li>
<li>Merge Qt translations into app applications as well</li>
</ul>
<p>With contributions from:</p>
<p>Heimen Stoffels, albanobattistella, mmahhi</p>
</description>
</release>
<release type="stable" version="0.9.2" date="2025-03-05T00:00:00Z">
<description>
<p>This release fixes the following bug:</p>
<ul>
<li>Refer to the right exiv2 CMake module so it can be found on Linux</li>
</ul>
<p>This release includes the following changes:</p>
<ul>
<li>Convert DEP5 to REUSE.toml for better REUSE compliance</li>
<li>Update translations</li>
</ul>
<p>With contributions from:</p>
<p>Pino Toscano, TamilNeram</p>
</description>
</release>
<release type="stable" version="0.9.1" date="2025-01-25T00:00:00Z">
<description>
<p>This release adds the following features:</p>
<ul>
<li>Option to double-click to fullscreen</li>
<li>Build-time option to embed translation resources</li>
</ul>
<p>This release fixes the following bugs:</p>
<ul>
<li>Fix window size not adjusted when open file on macOS</li>
<li>Should center window according to available screen geometry</li>
</ul>
<p>This release includes the following changes:</p>
<ul>
<li>Change close window bahavior on macOS</li>
<li>Update translations</li>
</ul>
<p>With contributions from:</p>
<p>albanobattistella, Sabri Ünal</p>
</description>
</release>
<release type="stable" version="0.9.0" date="2024-12-08T00:00:00Z">
<description>
<p>This release adds the following features:</p>
<ul>
<li>Support custom shortcuts for existing actions</li>
<li>Actions for frame-by-frame animated image playback support</li>
</ul>
<p>This release includes the following changes:</p>
<ul>
<li>Initial macOS bundle support</li>
<li>bump minimum required CMake version to 3.16</li>
<li>Update translations</li>
</ul>
<p>With contributions from:</p>
<p>albanobattistella, VenusGirl, gallegonovato, Sabri Ünal</p>
</description>
</release>
<release type="stable" version="0.8.2.1" date="2024-10-27T00:00:00Z">
<description>
<p>This release fixes the following bug:</p>
<ul>
<li>Cannot load translations caused by a change in 0.8.2</li>
</ul>
</description>
</release>
<release type="stable" version="0.8.2" date="2024-10-26T00:00:00Z">
<description>
<p>This release adds the following feature:</p>
<ul>
<li>New option to allow use light-color checkerboard by default</li>
</ul>
<p>With contributions from:</p>
<p>albanobattistella, mmahhi, gallegonovato</p>
</description>
</release>
<release type="stable" version="0.8.1" date="2024-08-25T00:00:00Z">
<description>
<p>This release adds the following feature:</p>
<ul>
<li>New command line option to list all supported formats</li>
</ul>
<p>With contributions from:</p>
<p>albanobattistella, mmahhi, ovl-1, gallegonovato, Oğuz Ersen</p>
</description>
</release>
<release type="stable" version="0.8.0" date="2024-06-29T00:00:00Z">
<description>
<p>This release adds the following feature:</p>
<ul>
<li>Support move image file to trash</li>
</ul>
<p>With contributions from:</p>
<p>albanobattistella, mmahhi, gallegonovato, Oğuz Ersen</p>
</description>
</release>
<release type="stable" version="0.7.4" date="2024-04-04T00:00:00Z">
<description>
<p>This release adds the following feature:</p>
<ul>
<li>Add some icons for corresponding menu actions</li>
</ul>
<p>With contributions from:</p>
<p>Reza Almanda, mmahhi, Oğuz Ersen, volkov, Сергій</p>
</description>
</release>
<release type="stable" version="0.7.3" date="2023-10-24T00:00:00Z">
<description>
<p>This release adds the following feature:</p>
<ul>
<li>Add "Keep transformation" to menu</li>
</ul>
<p>With contributions from:</p>
<p>mmahhi, VenusGirl, albanobattistella, gallegonovato, Heimen Stoffels</p>
</description>
</release>
<release type="stable" version="0.7.2" date="2023-08-27T00:00:00Z">
<description>
<p>This release adds the following feature:</p>
<ul>
<li>Add an option in setting dialog to tweak the High-DPI scaling rounding policy (might only works in Qt 6 build)</li>
</ul>
<p>This release fixes the following bugs:</p>
<ul>
<li>Remove image size limit for Qt 6 build</li>
<li>Fix application icon install location under Linux</li>
</ul>
<p>With contributions from:</p>
<p>Heimen Stoffels, Andrey, Dan, gallegonovato, albanobattistella, Sabri Ünal</p>
</description>
</release>
<release type="stable" version="0.7.1" date="2023-07-08T00:00:00Z">
<description>
<p>This release adds the following features:</p>
<ul>
<li>TIF and TIFF format files in the same folder will now be automatically added to the gallery</li>
<li>Built-in window resizing now also supports Linux desktop. (macOS might also works as well)</li>
</ul>
<p>This release fixes the following bugs:</p>
<ul>
<li>Settings dialog will automatedly use a suitable size instead of a hard-coded one</li>
<li>Fix default configuration file location under Linux. (was `~/.config/config.ini`, now it's `~/.config/Pineapple Pictures/config.ini`)</li>
</ul>
<p>With contributions from:</p>
<p>yyc12345</p>
</description>
</release>
</releases>
<content_rating type="oars-1.1"/> <content_rating type="oars-1.1"/>
<releases>
<release version="0.7.2" date="2023-08-27"/>
<release version="0.7.1" date="2023-07-08"/>
</releases>
</component> </component>

View File

@ -1,55 +0,0 @@
msgid ""
msgstr ""
"Project-Id-Version: PACKAGE VERSION\n"
"POT-Creation-Date: 2023-08-22 18:49中国标准时间\n"
"PO-Revision-Date: 2024-04-19 17:07+0000\n"
"Last-Translator: gallegonovato <fran-carro@hotmail.es>\n"
"Language-Team: Spanish <https://hosted.weblate.org/projects/pineapple-"
"pictures/appstream-metadata/es/>\n"
"Language: es\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.5-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 "Visor de imágenes"
#. (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 es un visor de imágenes ligero y fácil de usar que viene "
"con una práctica miniatura de navegación al hacer zoom, y no contiene ningún "
"soporte de gestión de imágenes."
#. (itstool) path: screenshot/caption
#: net.blumia.pineapple-pictures.metainfo.xml:17
msgid "Main window when an image file is loaded"
msgstr "Ventana principal cuando se carga un archivo de imagen"
#. (itstool) path: screenshot/caption
#: net.blumia.pineapple-pictures.metainfo.xml:22
msgid "Zooming in a raster image"
msgstr "Ampliar una imagen rasterizada"
#. (itstool) path: screenshot/caption
#: net.blumia.pineapple-pictures.metainfo.xml:27
msgid "Zooming in a vector image"
msgstr "Ampliar una imagen vectorial"
#. (itstool) path: component/developer_name
#: net.blumia.pineapple-pictures.metainfo.xml:34
msgid "Gary (BLumia) Wang et al."
msgstr "Gary (BLumia) Wang et al."

View File

@ -1,54 +0,0 @@
msgid ""
msgstr ""
"Project-Id-Version: PACKAGE VERSION\n"
"POT-Creation-Date: 2023-08-22 18:49中国标准时间\n"
"PO-Revision-Date: 2023-11-14 17:05+0000\n"
"Last-Translator: mmahhi <masa10suda@gmail.com>\n"
"Language-Team: Japanese <https://hosted.weblate.org/projects/pineapple-"
"pictures/appstream-metadata/ja/>\n"
"Language: ja\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
"Plural-Forms: nplurals=1; plural=0;\n"
"X-Generator: Weblate 5.2-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 など"

View File

@ -1,52 +0,0 @@
msgid ""
msgstr ""
"Project-Id-Version: PACKAGE VERSION\n"
"POT-Creation-Date: 2023-08-22 18:49中国标准时间\n"
"PO-Revision-Date: 2025-01-28 09:01+0000\n"
"Last-Translator: தமிழ்நேரம் <anishprabu.t@gmail.com>\n"
"Language-Team: Tamil <https://hosted.weblate.org/projects/pineapple-pictures/"
"appstream-metadata/ta/>\n"
"Language: ta\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.10-dev\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 "கேரி (ப்ளூமியா) வாங் மற்றும் பலர்."

View File

@ -1,56 +0,0 @@
msgid ""
msgstr ""
"Project-Id-Version: PACKAGE VERSION\n"
"POT-Creation-Date: 2023-08-22 18:49中国标准时间\n"
"PO-Revision-Date: 2024-01-01 16:10+0000\n"
"Last-Translator: Сергій <sergiy.goncharuk.1@gmail.com>\n"
"Language-Team: Ukrainian <https://hosted.weblate.org/projects/"
"pineapple-pictures/appstream-metadata/uk/>\n"
"Language: uk\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.4-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."

View File

@ -1,101 +1,93 @@
# SPDX-FileCopyrightText: 2025 Gary Wang <git@blumia.net> # SPDX-FileCopyrightText: 2022 Gary Wang <wzc782970009@gmail.com>
# #
# SPDX-License-Identifier: MIT # SPDX-License-Identifier: MIT
QT += core widgets gui svg svgwidgets QT += core widgets gui svg
greaterThan(QT_MAJOR_VERSION, 5): QT += svgwidgets
TARGET = ppic
TEMPLATE = app TARGET = ppic
DEFINES += PPIC_VERSION_STRING=\\\"x.y.z\\\" TEMPLATE = app
win32 { # The following define makes your compiler emit warnings if you use
DEFINES += FLAG_PORTABLE_MODE_SUPPORT=1 # 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.
# The following define makes your compiler emit warnings if you use DEFINES += QT_DEPRECATED_WARNINGS
# any feature of Qt which has been marked as deprecated (the exact warnings
# depend on your compiler). Please consult the documentation of the # You can also make your code fail to compile if you use deprecated APIs.
# deprecated API in order to know how to port your code away from it. # In order to do so, uncomment the following line.
DEFINES += QT_DEPRECATED_WARNINGS # 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
# You can also make your code fail to compile if you use deprecated APIs.
# In order to do so, uncomment the following line. CONFIG += c++11 lrelease embed_translations
# 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 SOURCES += \
app/aboutdialog.cpp \
CONFIG += c++17 lrelease embed_translations app/main.cpp \
app/framelesswindow.cpp \
SOURCES += \ app/mainwindow.cpp \
app/aboutdialog.cpp \ app/graphicsview.cpp \
app/main.cpp \ app/bottombuttongroup.cpp \
app/framelesswindow.cpp \ app/graphicsscene.cpp \
app/mainwindow.cpp \ app/navigatorview.cpp \
app/graphicsview.cpp \ app/opacityhelper.cpp \
app/bottombuttongroup.cpp \ app/toolbutton.cpp \
app/graphicsscene.cpp \ app/settings.cpp \
app/navigatorview.cpp \ app/settingsdialog.cpp \
app/opacityhelper.cpp \ app/metadatamodel.cpp \
app/toolbutton.cpp \ app/metadatadialog.cpp \
app/settings.cpp \ app/exiv2wrapper.cpp \
app/settingsdialog.cpp \ app/actionmanager.cpp \
app/metadatamodel.cpp \ app/playlistmanager.cpp
app/metadatadialog.cpp \
app/exiv2wrapper.cpp \ HEADERS += \
app/actionmanager.cpp \ app/aboutdialog.h \
app/playlistmanager.cpp \ app/framelesswindow.h \
app/shortcutedit.cpp \ app/mainwindow.h \
app/fileopeneventhandler.cpp app/graphicsview.h \
app/bottombuttongroup.h \
HEADERS += \ app/graphicsscene.h \
app/aboutdialog.h \ app/navigatorview.h \
app/framelesswindow.h \ app/opacityhelper.h \
app/mainwindow.h \ app/toolbutton.h \
app/graphicsview.h \ app/settings.h \
app/bottombuttongroup.h \ app/settingsdialog.h \
app/graphicsscene.h \ app/metadatamodel.h \
app/navigatorview.h \ app/metadatadialog.h \
app/opacityhelper.h \ app/exiv2wrapper.h \
app/toolbutton.h \ app/actionmanager.h \
app/settings.h \ app/playlistmanager.h
app/settingsdialog.h \
app/metadatamodel.h \ TRANSLATIONS = \
app/metadatadialog.h \ app/translations/PineapplePictures.ts \
app/exiv2wrapper.h \ app/translations/PineapplePictures_zh_CN.ts \
app/actionmanager.h \ app/translations/PineapplePictures_de.ts \
app/playlistmanager.h \ app/translations/PineapplePictures_es.ts \
app/shortcutedit.h \ app/translations/PineapplePictures_fr.ts \
app/fileopeneventhandler.h app/translations/PineapplePictures_nb_NO.ts \
app/translations/PineapplePictures_nl.ts \
TRANSLATIONS = \ app/translations/PineapplePictures_ru.ts \
app/translations/PineapplePictures_en.ts \ app/translations/PineapplePictures_si.ts \
app/translations/PineapplePictures_zh_CN.ts \ app/translations/PineapplePictures_id.ts
app/translations/PineapplePictures_de.ts \
app/translations/PineapplePictures_es.ts \ # Default rules for deployment.
app/translations/PineapplePictures_fr.ts \ qnx: target.path = /tmp/$${TARGET}/bin
app/translations/PineapplePictures_nb_NO.ts \ else: unix:!android: target.path = /opt/$${TARGET}/bin
app/translations/PineapplePictures_nl.ts \ !isEmpty(target.path): INSTALLS += target
app/translations/PineapplePictures_ru.ts \
app/translations/PineapplePictures_si.ts \ RESOURCES += \
app/translations/PineapplePictures_id.ts assets/resources.qrc
# Default rules for deployment. # Generate from svg:
qnx: target.path = /tmp/$${TARGET}/bin # magick convert -density 512x512 -background none app-icon.svg -define icon:auto-resize app-icon.ico
else: unix:!android: target.path = /opt/$${TARGET}/bin RC_ICONS = assets/icons/app-icon.ico
!isEmpty(target.path): INSTALLS += target
# Windows only, for rc file (we're not going to use the .rc file in this repo)
RESOURCES += \ QMAKE_TARGET_PRODUCT = Pineapple Pictures
assets/resources.qrc QMAKE_TARGET_DESCRIPTION = Pineapple Pictures - Image Viewer
QMAKE_TARGET_COPYRIGHT = MIT/Expat License - Copyright (C) 2020 Gary Wang
# Generate from svg:
# magick convert -density 512x512 -background none app-icon.svg -define icon:auto-resize app-icon.ico # MSVC only, since QMake doesn't have a CMAKE_CXX_STANDARD_LIBRARIES or cpp_winlibs similar thing
RC_ICONS = assets/icons/app-icon.ico win32-msvc* {
LIBS += -luser32
# 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) 2024 Gary Wang
# MSVC only, since QMake doesn't have a CMAKE_CXX_STANDARD_LIBRARIES or cpp_winlibs similar thing
win32-msvc* {
LIBS += -luser32
}