Merge branch 'develop' of https://github.com/PrismLauncher/PrismLauncher into import
Signed-off-by: Trial97 <alexandru.tripon97@gmail.com>
This commit is contained in:
		
							
								
								
									
										8
									
								
								.editorconfig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								.editorconfig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| # EditorConfig specs and documentation: https://EditorConfig.org | ||||
|  | ||||
| # top-most EditorConfig file | ||||
| root = true | ||||
|  | ||||
| # C++ Code Style settings | ||||
| [*.{c++,cc,cpp,cppm,cxx,h,h++,hh,hpp,hxx,inl,ipp,ixx,tlh,tli}] | ||||
| cpp_generate_documentation_comments = doxygen_slash_star | ||||
							
								
								
									
										32
									
								
								.github/workflows/backport.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								.github/workflows/backport.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| name: Backport | ||||
| on: | ||||
|   pull_request_target: | ||||
|     types: [closed, labeled] | ||||
|  | ||||
| # WARNING: | ||||
| # When extending this action, be aware that $GITHUB_TOKEN allows write access to | ||||
| # the GitHub repository. This means that it should not evaluate user input in a | ||||
| # way that allows code injection. | ||||
|  | ||||
| permissions: | ||||
|   contents: read | ||||
|  | ||||
| jobs: | ||||
|   backport: | ||||
|     permissions: | ||||
|       contents: write # for korthout/backport-action to create branch | ||||
|       pull-requests: write # for korthout/backport-action to create PR to backport | ||||
|     name: Backport Pull Request | ||||
|     if: github.repository_owner == 'PrismLauncher' && github.event.pull_request.merged == true && (github.event_name != 'labeled' || startsWith('backport', github.event.label.name)) | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v3 | ||||
|         with: | ||||
|           ref: ${{ github.event.pull_request.head.sha }} | ||||
|       - name: Create backport PRs | ||||
|         uses: korthout/backport-action@v1.3.1 | ||||
|         with: | ||||
|           # Config README: https://github.com/korthout/backport-action#backport-action | ||||
|           pull_description: |- | ||||
|             Bot-based backport to `${target_branch}`, triggered by a label in #${pull_number}. | ||||
|  | ||||
							
								
								
									
										10
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -264,23 +264,23 @@ jobs: | ||||
|       - name: Configure CMake (macOS) | ||||
|         if: runner.os == 'macOS' && matrix.qt_ver == 6 | ||||
|         run: | | ||||
|           cmake -S . -B ${{ env.BUILD_DIR }} -DCMAKE_INSTALL_PREFIX=${{ env.INSTALL_DIR }} -DCMAKE_BUILD_TYPE=${{ inputs.build_type }} -DENABLE_LTO=ON -DLauncher_BUILD_PLATFORM=${{ matrix.name }} -DCMAKE_C_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DCMAKE_CXX_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DLauncher_QT_VERSION_MAJOR=${{ matrix.qt_ver }} -DCMAKE_OSX_ARCHITECTURES="x86_64;arm64" -G Ninja | ||||
|           cmake -S . -B ${{ env.BUILD_DIR }} -DCMAKE_INSTALL_PREFIX=${{ env.INSTALL_DIR }} -DCMAKE_BUILD_TYPE=${{ inputs.build_type }} -DENABLE_LTO=ON -DLauncher_BUILD_PLATFORM=official -DCMAKE_C_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DCMAKE_CXX_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DLauncher_QT_VERSION_MAJOR=${{ matrix.qt_ver }} -DCMAKE_OSX_ARCHITECTURES="x86_64;arm64" -G Ninja | ||||
|  | ||||
|       - name: Configure CMake (macOS-Legacy) | ||||
|         if: runner.os == 'macOS' && matrix.qt_ver == 5 | ||||
|         run: | | ||||
|           cmake -S . -B ${{ env.BUILD_DIR }} -DCMAKE_INSTALL_PREFIX=${{ env.INSTALL_DIR }} -DCMAKE_BUILD_TYPE=${{ inputs.build_type }} -DENABLE_LTO=ON -DLauncher_BUILD_PLATFORM=${{ matrix.name }} -DCMAKE_C_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DCMAKE_CXX_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DLauncher_QT_VERSION_MAJOR=${{ matrix.qt_ver }} -DMACOSX_SPARKLE_UPDATE_PUBLIC_KEY="" -DMACOSX_SPARKLE_UPDATE_FEED_URL="" -G Ninja | ||||
|           cmake -S . -B ${{ env.BUILD_DIR }} -DCMAKE_INSTALL_PREFIX=${{ env.INSTALL_DIR }} -DCMAKE_BUILD_TYPE=${{ inputs.build_type }} -DENABLE_LTO=ON -DLauncher_BUILD_PLATFORM=official -DCMAKE_C_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DCMAKE_CXX_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DLauncher_QT_VERSION_MAJOR=${{ matrix.qt_ver }} -DMACOSX_SPARKLE_UPDATE_PUBLIC_KEY="" -DMACOSX_SPARKLE_UPDATE_FEED_URL="" -G Ninja | ||||
|  | ||||
|       - name: Configure CMake (Windows MinGW-w64) | ||||
|         if: runner.os == 'Windows' && matrix.msystem != '' | ||||
|         shell: msys2 {0} | ||||
|         run: | | ||||
|           cmake -S . -B ${{ env.BUILD_DIR }} -DCMAKE_INSTALL_PREFIX=${{ env.INSTALL_DIR }} -DCMAKE_BUILD_TYPE=${{ inputs.build_type }} -DENABLE_LTO=ON -DLauncher_BUILD_PLATFORM=${{ matrix.name }} -DCMAKE_C_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DCMAKE_CXX_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DLauncher_QT_VERSION_MAJOR=6 -DCMAKE_OBJDUMP=/mingw64/bin/objdump.exe -G Ninja | ||||
|           cmake -S . -B ${{ env.BUILD_DIR }} -DCMAKE_INSTALL_PREFIX=${{ env.INSTALL_DIR }} -DCMAKE_BUILD_TYPE=${{ inputs.build_type }} -DENABLE_LTO=ON -DLauncher_BUILD_PLATFORM=official -DCMAKE_C_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DCMAKE_CXX_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DLauncher_QT_VERSION_MAJOR=6 -DCMAKE_OBJDUMP=/mingw64/bin/objdump.exe -G Ninja | ||||
|  | ||||
|       - name: Configure CMake (Windows MSVC) | ||||
|         if: runner.os == 'Windows' && matrix.msystem == '' | ||||
|         run: | | ||||
|           cmake -S . -B ${{ env.BUILD_DIR }} -DCMAKE_INSTALL_PREFIX=${{ env.INSTALL_DIR }} -DCMAKE_BUILD_TYPE=${{ inputs.build_type }} -DENABLE_LTO=ON -DLauncher_BUILD_PLATFORM=${{ matrix.name }} -DLauncher_QT_VERSION_MAJOR=${{ matrix.qt_ver }} -DCMAKE_MSVC_RUNTIME_LIBRARY="MultiThreadedDLL" -A${{ matrix.architecture}} -DLauncher_FORCE_BUNDLED_LIBS=ON | ||||
|           cmake -S . -B ${{ env.BUILD_DIR }} -DCMAKE_INSTALL_PREFIX=${{ env.INSTALL_DIR }} -DCMAKE_BUILD_TYPE=${{ inputs.build_type }} -DENABLE_LTO=ON -DLauncher_BUILD_PLATFORM=official -DLauncher_QT_VERSION_MAJOR=${{ matrix.qt_ver }} -DCMAKE_MSVC_RUNTIME_LIBRARY="MultiThreadedDLL" -A${{ matrix.architecture}} -DLauncher_FORCE_BUNDLED_LIBS=ON | ||||
|           # https://github.com/ccache/ccache/wiki/MS-Visual-Studio (I coudn't figure out the compiler prefix) | ||||
|           if ("${{ env.CCACHE_VAR }}") | ||||
|           { | ||||
| @@ -295,7 +295,7 @@ jobs: | ||||
|       - name: Configure CMake (Linux) | ||||
|         if: runner.os == 'Linux' | ||||
|         run: | | ||||
|           cmake -S . -B ${{ env.BUILD_DIR }} -DCMAKE_INSTALL_PREFIX=/usr -DCMAKE_BUILD_TYPE=${{ inputs.build_type }} -DENABLE_LTO=ON -DLauncher_BUILD_PLATFORM=Linux -DCMAKE_C_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DCMAKE_CXX_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DLauncher_QT_VERSION_MAJOR=${{ matrix.qt_ver }} -G Ninja | ||||
|           cmake -S . -B ${{ env.BUILD_DIR }} -DCMAKE_INSTALL_PREFIX=/usr -DCMAKE_BUILD_TYPE=${{ inputs.build_type }} -DENABLE_LTO=ON -DLauncher_BUILD_PLATFORM=official -DCMAKE_C_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DCMAKE_CXX_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DLauncher_QT_VERSION_MAJOR=${{ matrix.qt_ver }} -G Ninja | ||||
|  | ||||
|       ## | ||||
|       # BUILD | ||||
|   | ||||
							
								
								
									
										3
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							| @@ -19,3 +19,6 @@ | ||||
| [submodule "libraries/cmark"] | ||||
| 	path = libraries/cmark | ||||
| 	url = https://github.com/commonmark/cmark.git | ||||
| [submodule "flatpak/shared-modules"] | ||||
| 	path = flatpak/shared-modules | ||||
| 	url = https://github.com/flathub/shared-modules.git | ||||
|   | ||||
| @@ -85,6 +85,38 @@ set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -DTOML_ENABLE_FLOAT16=0") | ||||
| # set CXXFLAGS for build targets | ||||
| set(CMAKE_CXX_FLAGS_RELEASE "-O2 -D_FORTIFY_SOURCE=2 ${CMAKE_CXX_FLAGS_RELEASE}") | ||||
|  | ||||
| option(DEBUG_ADDRESS_SANITIZER "Enable Address Sanitizer in Debug builds" on) | ||||
|  | ||||
| # If this is a Debug build turn on address sanitiser | ||||
| if (CMAKE_BUILD_TYPE STREQUAL "Debug" AND DEBUG_ADDRESS_SANITIZER) | ||||
|     message(STATUS "Address Sanitizer enabled for Debug builds, Turn it off with -DDEBUG_ADDRESS_SANITIZER=off") | ||||
|     if ("${CMAKE_CXX_COMPILER_ID}" MATCHES "Clang") | ||||
|         if (CMAKE_CXX_COMPILER_FRONTEND_VARIANT STREQUAL "MSVC") | ||||
|             # using clang with clang-cl front end  | ||||
|             message(STATUS "Address Sanitizer available on Clang MSVC frontend") | ||||
|             set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /fsanitize=address /O1 /Oy-") | ||||
|             set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} /fsanitize=address /O1 /Oy-") | ||||
|         else() | ||||
|             # AppleClang and Clang | ||||
|             message(STATUS "Address Sanitizer available on Clang") | ||||
|             set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=address -O1 -fno-omit-frame-pointer") | ||||
|             set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fsanitize=address -O1 -fno-omit-frame-pointer") | ||||
|         endif() | ||||
|     elseif ("${CMAKE_CXX_COMPILER_ID}" STREQUAL "GNU") | ||||
|         # GCC | ||||
|         message(STATUS "Address Sanitizer available on GCC") | ||||
|         set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=address -O1 -fno-omit-frame-pointer") | ||||
|         set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fsanitize=address -O1 -fno-omit-frame-pointer") | ||||
|         link_libraries("asan") | ||||
|     elseif ("${CMAKE_CXX_COMPILER_ID}" STREQUAL "MSVC") | ||||
|         message(STATUS "Address Sanitizer available on MSVC") | ||||
|         set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /fsanitize=address /O1 /Oy-") | ||||
|         set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} /fsanitize=address /O1 /Oy-") | ||||
|     else() | ||||
|         message(STATUS "Address Sanitizer not available on compiler ${CMAKE_CXX_COMPILER_ID}") | ||||
|     endif() | ||||
| endif() | ||||
|  | ||||
| option(ENABLE_LTO "Enable Link Time Optimization" off) | ||||
|  | ||||
| if(ENABLE_LTO) | ||||
| @@ -146,7 +178,7 @@ set(Launcher_VERSION_NAME4 "${Launcher_VERSION_MAJOR}.${Launcher_VERSION_MINOR}. | ||||
| set(Launcher_VERSION_NAME4_COMMA "${Launcher_VERSION_MAJOR},${Launcher_VERSION_MINOR},0,0") | ||||
|  | ||||
| # Build platform. | ||||
| set(Launcher_BUILD_PLATFORM "" CACHE STRING "A short string identifying the platform that this build was built for. Only used to display in the about dialog.") | ||||
| set(Launcher_BUILD_PLATFORM "unknown" CACHE STRING "A short string identifying the platform that this build was built for. Only used to display in the about dialog.") | ||||
|  | ||||
| # Channel list URL | ||||
| set(Launcher_UPDATER_BASE "" CACHE STRING "Base URL for the updater.") | ||||
| @@ -332,7 +364,7 @@ elseif(UNIX) | ||||
|  | ||||
|     set(BINARY_DEST_DIR "bin") | ||||
|     set(LIBRARY_DEST_DIR "lib${LIB_SUFFIX}") | ||||
|     set(JARS_DEST_DIR "share/${Launcher_APP_BINARY_NAME}") | ||||
|     set(JARS_DEST_DIR "share/${Launcher_Name}") | ||||
|  | ||||
|     # install as bundle with no dependencies included | ||||
|     set(INSTALL_BUNDLE "nodeps") | ||||
| @@ -345,7 +377,7 @@ elseif(UNIX) | ||||
|     install(FILES ${CMAKE_CURRENT_SOURCE_DIR}/${Launcher_SVG} DESTINATION "${KDE_INSTALL_ICONDIR}/hicolor/scalable/apps") | ||||
|     install(FILES ${CMAKE_CURRENT_SOURCE_DIR}/${Launcher_mrpack_MIMEInfo} DESTINATION ${KDE_INSTALL_MIMEDIR}) | ||||
|  | ||||
|     install(FILES "${CMAKE_CURRENT_SOURCE_DIR}/launcher/qtlogging.ini" DESTINATION "${KDE_INSTALL_DATADIR}/${Launcher_Name}") | ||||
|     install(FILES "${CMAKE_CURRENT_SOURCE_DIR}/launcher/qtlogging.ini" DESTINATION "share/${Launcher_Name}") | ||||
|      | ||||
|     if(Launcher_ManPage) | ||||
|         install(FILES ${CMAKE_CURRENT_BINARY_DIR}/${Launcher_ManPage} DESTINATION "${KDE_INSTALL_MANDIR}/man6") | ||||
|   | ||||
| @@ -61,3 +61,10 @@ As a bonus, you can also [cryptographically sign your commits][gh-signing-commit | ||||
|  | ||||
| [gh-signing-commits]: https://docs.github.com/en/authentication/managing-commit-signature-verification/signing-commits | ||||
| [gh-vigilant-mode]: https://docs.github.com/en/authentication/managing-commit-signature-verification/displaying-verification-statuses-for-all-of-your-commits | ||||
|  | ||||
|  | ||||
| ## Backporting to Release Branches | ||||
|  | ||||
| We use [automated backports](https://github.com/PrismLauncher/PrismLauncher/blob/develop/.github/workflows/backport.yml) to merge specific contributions from develop into `release` branches. | ||||
|  | ||||
| This is done when pull requests are merged and have labels such as `backport release-7.x` - which should be added along with the milestone for the release. | ||||
|   | ||||
| @@ -65,7 +65,7 @@ Config::Config() | ||||
|     MAC_SPARKLE_PUB_KEY = "@MACOSX_SPARKLE_UPDATE_PUBLIC_KEY@"; | ||||
|     MAC_SPARKLE_APPCAST_URL = "@MACOSX_SPARKLE_UPDATE_FEED_URL@"; | ||||
|  | ||||
|     if (BUILD_PLATFORM == "macOS" && !MAC_SPARKLE_PUB_KEY.isEmpty() && !MAC_SPARKLE_APPCAST_URL.isEmpty()) | ||||
|     if (!MAC_SPARKLE_PUB_KEY.isEmpty() && !MAC_SPARKLE_APPCAST_URL.isEmpty()) | ||||
|     { | ||||
|         UPDATER_ENABLED = true; | ||||
|     } | ||||
|   | ||||
| @@ -68,7 +68,7 @@ class Config { | ||||
|  | ||||
|     bool UPDATER_ENABLED = false; | ||||
|  | ||||
|     /// A short string identifying this build's platform. For example, "lin64" or "win32". | ||||
|     /// A short string identifying this build's platform or distribution. | ||||
|     QString BUILD_PLATFORM; | ||||
|  | ||||
|     /// A string containing the build timestamp | ||||
|   | ||||
							
								
								
									
										22
									
								
								flatpak/libdecor.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								flatpak/libdecor.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| { | ||||
|   "name": "libdecor", | ||||
|   "buildsystem": "meson", | ||||
|   "config-opts": [ | ||||
|     "-Ddemo=false" | ||||
|   ], | ||||
|   "sources": [ | ||||
|     { | ||||
|       "type": "git", | ||||
|       "url": "https://gitlab.freedesktop.org/libdecor/libdecor.git", | ||||
|       "commit": "73260393a97291c887e1074ab7f318e031be0ac6" | ||||
|     }, | ||||
|     { | ||||
|       "type": "patch", | ||||
|       "path": "patches/weird_libdecor.patch" | ||||
|     } | ||||
|   ], | ||||
|   "cleanup": [ | ||||
|     "/include", | ||||
|     "/lib/pkgconfig" | ||||
|   ] | ||||
| } | ||||
| @@ -5,13 +5,6 @@ sdk: org.kde.Sdk | ||||
| sdk-extensions: | ||||
|   - org.freedesktop.Sdk.Extension.openjdk17 | ||||
|   - org.freedesktop.Sdk.Extension.openjdk8 | ||||
| add-extensions: | ||||
|   com.valvesoftware.Steam.Utility.gamescope: | ||||
|     version: stable | ||||
|     add-ld-path: lib | ||||
|     no-autodownload: true | ||||
|     autodelete: false | ||||
|     directory: utils/gamescope | ||||
|  | ||||
| command: prismlauncher | ||||
| finish-args: | ||||
| @@ -28,12 +21,22 @@ finish-args: | ||||
|     # FTBApp import | ||||
|   - --filesystem=~/.ftba:ro | ||||
|  | ||||
| cleanup: | ||||
|   - /lib/libGLU* | ||||
|  | ||||
| modules: | ||||
|   # Might be needed by some Controller mods (see https://github.com/isXander/Controlify/issues/31) | ||||
|   - shared-modules/libusb/libusb.json | ||||
|  | ||||
|   # Needed for proper Wayland support | ||||
|   - libdecor.json | ||||
|  | ||||
|   - name: prismlauncher | ||||
|     buildsystem: cmake-ninja | ||||
|     builddir: true | ||||
|     config-opts: | ||||
|       - -DLauncher_BUILD_PLATFORM=flatpak | ||||
|       - -DCMAKE_BUILD_TYPE=Debug | ||||
|       - -DCMAKE_BUILD_TYPE=RelWithDebInfo | ||||
|       - -DLauncher_QT_VERSION_MAJOR=5 | ||||
|     build-options: | ||||
|       env: | ||||
| @@ -42,7 +45,7 @@ modules: | ||||
|     sources: | ||||
|       - type: dir | ||||
|         path: ../ | ||||
|     builddir: true | ||||
|  | ||||
|   - name: openjdk | ||||
|     buildsystem: simple | ||||
|     build-commands: | ||||
| @@ -51,14 +54,45 @@ modules: | ||||
|       - mv /app/jre /app/jdk/17 | ||||
|       - /usr/lib/sdk/openjdk8/install.sh | ||||
|       - mv /app/jre /app/jdk/8 | ||||
|     cleanup: [/jre] | ||||
|     cleanup: | ||||
|       - /jre | ||||
|  | ||||
|   - name: glfw | ||||
|     buildsystem: cmake-ninja | ||||
|     config-opts: | ||||
|       - -DCMAKE_BUILD_TYPE=RelWithDebInfo | ||||
|       - -DBUILD_SHARED_LIBS:BOOL=ON | ||||
|       - -DGLFW_USE_WAYLAND=ON | ||||
|     sources: | ||||
|       - type: git | ||||
|         url: https://github.com/glfw/glfw.git | ||||
|         commit: 3fa2360720eeba1964df3c0ecf4b5df8648a8e52 | ||||
|       - type: patch | ||||
|         path: patches/0003-Don-t-crash-on-calls-to-focus-or-icon.patch | ||||
|       - type: patch | ||||
|         path: patches/0005-Add-warning-about-being-an-unofficial-patch.patch | ||||
|       - type: patch | ||||
|         path: patches/0007-Platform-Prefer-Wayland-over-X11.patch | ||||
|     cleanup: | ||||
|       - /include | ||||
|       - /lib/cmake | ||||
|       - /lib/pkgconfig | ||||
|  | ||||
|   - name: xrandr | ||||
|     buildsystem: autotools | ||||
|     sources: | ||||
|       - type: archive | ||||
|         url: https://xorg.freedesktop.org/archive/individual/app/xrandr-1.5.1.tar.xz | ||||
|         sha256: 7bc76daf9d72f8aff885efad04ce06b90488a1a169d118dea8a2b661832e8762 | ||||
|     cleanup: [/share/man, /bin/xkeystone] | ||||
|         url: https://xorg.freedesktop.org/archive/individual/app/xrandr-1.5.2.tar.xz | ||||
|         sha256: c8bee4790d9058bacc4b6246456c58021db58a87ddda1a9d0139bf5f18f1f240 | ||||
|         x-checker-data: | ||||
|           type: anitya | ||||
|           project-id: 14957 | ||||
|           stable-only: true | ||||
|           url-template: https://xorg.freedesktop.org/archive/individual/app/xrandr-$version.tar.xz | ||||
|     cleanup: | ||||
|       - /share/man | ||||
|       - /bin/xkeystone | ||||
|  | ||||
|   - name: gamemode | ||||
|     buildsystem: meson | ||||
|     config-opts: | ||||
| @@ -69,19 +103,56 @@ modules: | ||||
|       # post-install is running inside the build dir, we need it from the source though | ||||
|       - install -Dm755 ../data/gamemoderun -t /app/bin | ||||
|     sources: | ||||
|       - type: git | ||||
|         url: https://github.com/FeralInteractive/gamemode | ||||
|         tag: "1.7" | ||||
|         commit: 4dc99dff76218718763a6b07fc1900fa6d1dafd9 | ||||
|       - type: archive | ||||
|         archive-type: tar-gzip | ||||
|         url: https://api.github.com/repos/FeralInteractive/gamemode/tarball/1.7 | ||||
|         sha256: 57ce73ba605d1cf12f8d13725006a895182308d93eba0f69f285648449641803 | ||||
|         x-checker-data: | ||||
|           type: json | ||||
|           url: https://api.github.com/repos/FeralInteractive/gamemode/releases/latest | ||||
|           version-query: .tag_name | ||||
|           url-query: .tarball_url | ||||
|           timestamp-query: .published_at | ||||
|     cleanup: | ||||
|       - /include | ||||
|       - /lib/pkgconfig | ||||
|       - /lib/libgamemodeauto.a | ||||
|  | ||||
|   - name: glxinfo | ||||
|     buildsystem: meson | ||||
|     config-opts: | ||||
|       - --bindir=/app/mesa-demos | ||||
|       - -Degl=disabled | ||||
|       - -Dglut=disabled | ||||
|       - -Dosmesa=disabled | ||||
|       - -Dvulkan=disabled | ||||
|       - -Dwayland=disabled | ||||
|     post-install: | ||||
|       - mv -v /app/mesa-demos/glxinfo /app/bin | ||||
|     sources: | ||||
|       - type: archive | ||||
|         url: https://archive.mesa3d.org/demos/mesa-demos-9.0.0.tar.xz | ||||
|         sha256: 3046a3d26a7b051af7ebdd257a5f23bfeb160cad6ed952329cdff1e9f1ed496b | ||||
|         x-checker-data: | ||||
|           type: anitya | ||||
|           project-id: 16781 | ||||
|           stable-only: true | ||||
|           url-template: https://archive.mesa3d.org/demos/mesa-demos-$version.tar.xz | ||||
|     cleanup: | ||||
|       - /include | ||||
|       - /mesa-demos | ||||
|       - /share | ||||
|     modules: | ||||
|       - shared-modules/glu/glu-9.json | ||||
|  | ||||
|   - name: enhance | ||||
|     buildsystem: simple | ||||
|     build-commands: | ||||
|       - mkdir -p /app/utils/gamescope | ||||
|       - install -Dm755 prime-run /app/bin/prime-run | ||||
|       - mv /app/bin/prismlauncher /app/bin/prismrun | ||||
|       - install -Dm755 prismlauncher /app/bin/prismlauncher | ||||
|     sources: | ||||
|       - type: file | ||||
|         path: ../flatpak/prime-run | ||||
|         path: prime-run | ||||
|       - type: file | ||||
|         path: ../flatpak/prismlauncher | ||||
|         path: prismlauncher | ||||
|   | ||||
| @@ -0,0 +1,24 @@ | ||||
| diff --git a/src/wl_window.c b/src/wl_window.c | ||||
| index 52d3b9eb..4ac4eb5d 100644 | ||||
| --- a/src/wl_window.c | ||||
| +++ b/src/wl_window.c | ||||
| @@ -2117,8 +2117,7 @@ void _glfwSetWindowTitleWayland(_GLFWwindow* window, const char* title) | ||||
|  void _glfwSetWindowIconWayland(_GLFWwindow* window, | ||||
|                                 int count, const GLFWimage* images) | ||||
|  { | ||||
| -    _glfwInputError(GLFW_FEATURE_UNAVAILABLE, | ||||
| -                    "Wayland: The platform does not support setting the window icon"); | ||||
| +    fprintf(stderr, "!!! Ignoring Error: Wayland: The platform does not support setting the window icon\n"); | ||||
|  } | ||||
|   | ||||
|  void _glfwGetWindowPosWayland(_GLFWwindow* window, int* xpos, int* ypos) | ||||
| @@ -2361,8 +2360,7 @@ void _glfwRequestWindowAttentionWayland(_GLFWwindow* window) | ||||
|   | ||||
|  void _glfwFocusWindowWayland(_GLFWwindow* window) | ||||
|  { | ||||
| -    _glfwInputError(GLFW_FEATURE_UNAVAILABLE, | ||||
| -                    "Wayland: The platform does not support setting the input focus"); | ||||
| +    fprintf(stderr, "!!! Ignoring Error: Wayland: The platform does not support setting the input focus\n"); | ||||
|  } | ||||
|   | ||||
|  void _glfwSetWindowMonitorWayland(_GLFWwindow* window, | ||||
| @@ -0,0 +1,17 @@ | ||||
| diff --git a/src/init.c b/src/init.c | ||||
| index 06dbb3f2..a7c6da86 100644 | ||||
| --- a/src/init.c | ||||
| +++ b/src/init.c | ||||
| @@ -449,6 +449,12 @@ GLFWAPI int glfwInit(void) | ||||
|      _glfw.initialized = GLFW_TRUE; | ||||
|   | ||||
|      glfwDefaultWindowHints(); | ||||
| + | ||||
| +    fprintf(stderr, "!!! Patched GLFW from https://github.com/Admicos/minecraft-wayland\n" | ||||
| +         "!!! If any issues with the window, or some issues with rendering, occur, " | ||||
| +         "first try with the built-in GLFW, and if that solves the issue, report there first.\n" | ||||
| +         "!!! Use outside Minecraft is untested, and things might break.\n"); | ||||
| + | ||||
|      return GLFW_TRUE; | ||||
|  } | ||||
|   | ||||
							
								
								
									
										20
									
								
								flatpak/patches/0007-Platform-Prefer-Wayland-over-X11.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								flatpak/patches/0007-Platform-Prefer-Wayland-over-X11.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| diff --git a/src/platform.c b/src/platform.c | ||||
| index c5966ae7..3e7442f9 100644 | ||||
| --- a/src/platform.c | ||||
| +++ b/src/platform.c | ||||
| @@ -49,12 +49,12 @@ static const struct | ||||
|  #if defined(_GLFW_COCOA) | ||||
|      { GLFW_PLATFORM_COCOA, _glfwConnectCocoa }, | ||||
|  #endif | ||||
| -#if defined(_GLFW_X11) | ||||
| -    { GLFW_PLATFORM_X11, _glfwConnectX11 }, | ||||
| -#endif | ||||
|  #if defined(_GLFW_WAYLAND) | ||||
|      { GLFW_PLATFORM_WAYLAND, _glfwConnectWayland }, | ||||
|  #endif | ||||
| +#if defined(_GLFW_X11) | ||||
| +    { GLFW_PLATFORM_X11, _glfwConnectX11 }, | ||||
| +#endif | ||||
|  }; | ||||
|   | ||||
|  GLFWbool _glfwSelectPlatform(int desiredID, _GLFWplatform* platform) | ||||
							
								
								
									
										40
									
								
								flatpak/patches/weird_libdecor.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								flatpak/patches/weird_libdecor.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | ||||
| diff --git a/src/libdecor.c b/src/libdecor.c | ||||
| index a9c1106..1aa38b3 100644 | ||||
| --- a/src/libdecor.c | ||||
| +++ b/src/libdecor.c | ||||
| @@ -1391,22 +1391,32 @@ calculate_priority(const struct libdecor_plugin_description *plugin_description) | ||||
|  static bool | ||||
|  check_symbol_conflicts(const struct libdecor_plugin_description *plugin_description) | ||||
|  { | ||||
| +	bool ret = true; | ||||
|  	char * const *symbol; | ||||
| +	void* main_prog = dlopen(NULL, RTLD_LAZY); | ||||
| +	if (!main_prog) { | ||||
| +		fprintf(stderr, "Plugin \"%s\" couldn't check conflicting symbols: \"%s\".\n", | ||||
| +					plugin_description->description, dlerror()); | ||||
| +		return false; | ||||
| +	} | ||||
| + | ||||
|   | ||||
|  	symbol = plugin_description->conflicting_symbols; | ||||
|  	while (*symbol) { | ||||
|  		dlerror(); | ||||
| -		dlsym (RTLD_DEFAULT, *symbol); | ||||
| +		dlsym (main_prog, *symbol); | ||||
|  		if (!dlerror()) { | ||||
|  			fprintf(stderr, "Plugin \"%s\" uses conflicting symbol \"%s\".\n", | ||||
|  					plugin_description->description, *symbol); | ||||
| -			return false; | ||||
| +			ret = false; | ||||
| +			break; | ||||
|  		} | ||||
|   | ||||
|  		symbol++; | ||||
|  	} | ||||
|   | ||||
| -	return true; | ||||
| +	dlclose(main_prog); | ||||
| +	return ret; | ||||
|  } | ||||
|   | ||||
|  static struct plugin_loader * | ||||
| @@ -5,7 +5,7 @@ for i in {0..9}; do | ||||
|     test -S "$XDG_RUNTIME_DIR"/discord-ipc-"$i" || ln -sf {app/com.discordapp.Discord,"$XDG_RUNTIME_DIR"}/discord-ipc-"$i"; | ||||
| done | ||||
|  | ||||
| export PATH="${PATH}${PATH:+:}/app/utils/gamescope/bin:/usr/lib/extensions/vulkan/MangoHud/bin" | ||||
| export LD_LIBRARY_PATH="${LD_LIBRARY_PATH}${LD_LIBRARY_PATH:+:}/usr/lib/extensions/vulkan/MangoHud/\$LIB/" | ||||
| export PATH="${PATH}${PATH:+:}/usr/lib/extensions/vulkan/gamescope/bin:/usr/lib/extensions/vulkan/MangoHud/bin" | ||||
| export VK_LAYER_PATH="/usr/lib/extensions/vulkan/share/vulkan/implicit_layer.d/" | ||||
|  | ||||
| exec /app/bin/prismrun "$@" | ||||
|   | ||||
							
								
								
									
										1
									
								
								flatpak/shared-modules
									
									
									
									
									
										Submodule
									
								
							
							
								
								
								
								
								
							
						
						
									
										1
									
								
								flatpak/shared-modules
									
									
									
									
									
										Submodule
									
								
							 Submodule flatpak/shared-modules added at 45094ca570
									
								
							| @@ -6,9 +6,10 @@ | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net> | ||||
|  *  Copyright (C) 2022 Lenny McLennington <lenny@sneed.church> | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  *  Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me> | ||||
|  *  Copyright (C) 2023 Rachel Powers <508861+Ryex@users.noreply.github.com> | ||||
|  *  Copyright (C) 2023 seth <getchoo at tuta dot io> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
| @@ -433,7 +434,11 @@ Application::Application(int &argc, char **argv) : QApplication(argc, argv) | ||||
|         } | ||||
|         // seach root path | ||||
|         if(!foundLoggingRules) { | ||||
| #if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD) || defined(Q_OS_OPENBSD) | ||||
|            logRulesPath = FS::PathCombine(m_rootPath, "share", BuildConfig.LAUNCHER_NAME, logRulesFile);  | ||||
| #else | ||||
|            logRulesPath = FS::PathCombine(m_rootPath, logRulesFile);  | ||||
| #endif | ||||
|             qDebug() << "Testing" << logRulesPath << "..."; | ||||
|             foundLoggingRules = QFile::exists(logRulesPath); | ||||
|         } | ||||
| @@ -471,6 +476,7 @@ Application::Application(int &argc, char **argv) : QApplication(argc, argv) | ||||
|  | ||||
|         qDebug() << BuildConfig.LAUNCHER_DISPLAYNAME << ", (c) 2013-2021 " << BuildConfig.LAUNCHER_COPYRIGHT; | ||||
|         qDebug() << "Version                    : " << BuildConfig.printableVersionString(); | ||||
|         qDebug() << "Platform                   : " << BuildConfig.BUILD_PLATFORM; | ||||
|         qDebug() << "Git commit                 : " << BuildConfig.GIT_COMMIT; | ||||
|         qDebug() << "Git refspec                : " << BuildConfig.GIT_REFSPEC; | ||||
|         if (adjustedBy.size()) | ||||
| @@ -568,6 +574,7 @@ Application::Application(int &argc, char **argv) : QApplication(argc, argv) | ||||
|  | ||||
|         // Language | ||||
|         m_settings->registerSetting("Language", QString()); | ||||
|         m_settings->registerSetting("UseSystemLocale", false); | ||||
|  | ||||
|         // Console | ||||
|         m_settings->registerSetting("ShowConsole", false); | ||||
| @@ -604,6 +611,9 @@ Application::Application(int &argc, char **argv) : QApplication(argc, argv) | ||||
|         m_settings->registerSetting("IgnoreJavaCompatibility", false); | ||||
|         m_settings->registerSetting("IgnoreJavaWizard", false); | ||||
|  | ||||
|         // Mod loader settings | ||||
|         m_settings->registerSetting("DisableQuiltBeacon", false); | ||||
|  | ||||
|         // Native library workarounds | ||||
|         m_settings->registerSetting("UseNativeOpenAL", false); | ||||
|         m_settings->registerSetting("UseNativeGLFW", false); | ||||
| @@ -918,12 +928,7 @@ bool Application::createSetupWizard() | ||||
|         } | ||||
|         return false; | ||||
|     }(); | ||||
|     bool languageRequired = [&]() | ||||
|     { | ||||
|         if (settings()->get("Language").toString().isEmpty()) | ||||
|             return true; | ||||
|         return false; | ||||
|     }(); | ||||
|     bool languageRequired = settings()->get("Language").toString().isEmpty(); | ||||
|     bool pasteInterventionRequired = settings()->get("PastebinURL") != ""; | ||||
|     bool themeInterventionRequired = settings()->get("ApplicationTheme") == ""; | ||||
|     bool wizardRequired = javaRequired || languageRequired || pasteInterventionRequired || themeInterventionRequired; | ||||
| @@ -1567,7 +1572,7 @@ QString Application::getJarPath(QString jarFile) | ||||
| { | ||||
|     QStringList potentialPaths = { | ||||
| #if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD) || defined(Q_OS_OPENBSD) | ||||
|         FS::PathCombine(m_rootPath, "share/" + BuildConfig.LAUNCHER_APP_BINARY_NAME), | ||||
|         FS::PathCombine(m_rootPath, "share", BuildConfig.LAUNCHER_NAME), | ||||
| #endif | ||||
|         FS::PathCombine(m_rootPath, "jars"), | ||||
|         FS::PathCombine(applicationDirPath(), "jars"), | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net> | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  *  Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|   | ||||
| @@ -15,16 +15,15 @@ | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include <memory> | ||||
| #include <QString> | ||||
| #include <QMetaType> | ||||
| #include <QString> | ||||
| #include <memory> | ||||
|  | ||||
| /*! | ||||
|  * An abstract base class for versions. | ||||
|  */ | ||||
| class BaseVersion | ||||
| { | ||||
| public: | ||||
| class BaseVersion { | ||||
|    public: | ||||
|     using Ptr = std::shared_ptr<BaseVersion>; | ||||
|     virtual ~BaseVersion() {} | ||||
|     /*! | ||||
| @@ -45,14 +44,8 @@ public: | ||||
|      */ | ||||
|     virtual QString typeString() const = 0; | ||||
|  | ||||
|     virtual bool operator<(BaseVersion &a) | ||||
|     { | ||||
|         return name() < a.name(); | ||||
|     }; | ||||
|     virtual bool operator>(BaseVersion &a) | ||||
|     { | ||||
|         return name() > a.name(); | ||||
|     }; | ||||
|     virtual bool operator<(BaseVersion& a) { return name() < a.name(); }; | ||||
|     virtual bool operator>(BaseVersion& a) { return name() > a.name(); }; | ||||
| }; | ||||
|  | ||||
| Q_DECLARE_METATYPE(BaseVersion::Ptr) | ||||
|   | ||||
| @@ -487,6 +487,9 @@ set(API_SOURCES | ||||
|     modplatform/helpers/HashUtils.cpp | ||||
|     modplatform/helpers/OverrideUtils.h | ||||
|     modplatform/helpers/OverrideUtils.cpp | ||||
|  | ||||
|     modplatform/helpers/ExportToModList.h | ||||
|     modplatform/helpers/ExportToModList.cpp | ||||
| ) | ||||
|  | ||||
| set(FTB_SOURCES | ||||
| @@ -921,6 +924,8 @@ SET(LAUNCHER_SOURCES | ||||
|     ui/dialogs/ExportInstanceDialog.h | ||||
|     ui/dialogs/ExportPackDialog.cpp | ||||
|     ui/dialogs/ExportPackDialog.h | ||||
|     ui/dialogs/ExportToModListDialog.cpp | ||||
|     ui/dialogs/ExportToModListDialog.h | ||||
|     ui/dialogs/IconPickerDialog.cpp | ||||
|     ui/dialogs/IconPickerDialog.h | ||||
|     ui/dialogs/ImportResourceDialog.cpp | ||||
| @@ -1069,6 +1074,7 @@ qt_wrap_ui(LAUNCHER_UI | ||||
|     ui/dialogs/SkinUploadDialog.ui | ||||
|     ui/dialogs/ExportInstanceDialog.ui | ||||
|     ui/dialogs/ExportPackDialog.ui | ||||
|     ui/dialogs/ExportToModListDialog.ui | ||||
|     ui/dialogs/IconPickerDialog.ui | ||||
|     ui/dialogs/ImportResourceDialog.ui | ||||
|     ui/dialogs/MSALoginDialog.ui | ||||
|   | ||||
| @@ -390,7 +390,10 @@ void LaunchController::launchInstance() | ||||
|     m_launcher->prependStep(makeShared<TextPrint>(m_launcher.get(), "Launched instance in " + online_mode + " mode\n", MessageLevel::Launcher)); | ||||
|  | ||||
|     // Prepend Version | ||||
|     m_launcher->prependStep(makeShared<TextPrint>(m_launcher.get(), BuildConfig.LAUNCHER_DISPLAYNAME + " version: " + BuildConfig.printableVersionString() + "\n\n", MessageLevel::Launcher)); | ||||
|     { | ||||
|         auto versionString = QString("%1 version: %2 (%3)").arg(BuildConfig.LAUNCHER_DISPLAYNAME, BuildConfig.printableVersionString(), BuildConfig.BUILD_PLATFORM); | ||||
|         m_launcher->prependStep(makeShared<TextPrint>(m_launcher.get(), versionString + "\n\n", MessageLevel::Launcher)); | ||||
|     } | ||||
|     m_launcher->start(); | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,7 +1,8 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  PolyMC - Minecraft Launcher | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net> | ||||
|  *  Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
| @@ -33,56 +34,50 @@ | ||||
|  *      limitations under the License. | ||||
|  */ | ||||
|  | ||||
| #include "MMCZip.h" | ||||
| #include <quazip/quazip.h> | ||||
| #include <quazip/quazipdir.h> | ||||
| #include <quazip/quazipfile.h> | ||||
| #include "MMCZip.h" | ||||
| #include "FileSystem.h" | ||||
|  | ||||
| #include <QCoreApplication> | ||||
| #include <QDebug> | ||||
| #include <QtConcurrentRun> | ||||
|  | ||||
| namespace MMCZip { | ||||
| // ours | ||||
| bool MMCZip::mergeZipFiles(QuaZip *into, QFileInfo from, QSet<QString> &contained, const FilterFunction filter) | ||||
| bool mergeZipFiles(QuaZip* into, QFileInfo from, QSet<QString>& contained, const FilterFunction filter) | ||||
| { | ||||
|     QuaZip modZip(from.filePath()); | ||||
|     modZip.open(QuaZip::mdUnzip); | ||||
|  | ||||
|     QuaZipFile fileInsideMod(&modZip); | ||||
|     QuaZipFile zipOutFile(into); | ||||
|     for (bool more = modZip.goToFirstFile(); more; more = modZip.goToNextFile()) | ||||
|     { | ||||
|     for (bool more = modZip.goToFirstFile(); more; more = modZip.goToNextFile()) { | ||||
|         QString filename = modZip.getCurrentFileName(); | ||||
|         if (filter && !filter(filename)) | ||||
|         { | ||||
|             qDebug() << "Skipping file " << filename << " from " | ||||
|                         << from.fileName() << " - filtered"; | ||||
|         if (filter && !filter(filename)) { | ||||
|             qDebug() << "Skipping file " << filename << " from " << from.fileName() << " - filtered"; | ||||
|             continue; | ||||
|         } | ||||
|         if (contained.contains(filename)) | ||||
|         { | ||||
|             qDebug() << "Skipping already contained file " << filename << " from " | ||||
|                         << from.fileName(); | ||||
|         if (contained.contains(filename)) { | ||||
|             qDebug() << "Skipping already contained file " << filename << " from " << from.fileName(); | ||||
|             continue; | ||||
|         } | ||||
|         contained.insert(filename); | ||||
|  | ||||
|         if (!fileInsideMod.open(QIODevice::ReadOnly)) | ||||
|         { | ||||
|         if (!fileInsideMod.open(QIODevice::ReadOnly)) { | ||||
|             qCritical() << "Failed to open " << filename << " from " << from.fileName(); | ||||
|             return false; | ||||
|         } | ||||
|  | ||||
|         QuaZipNewInfo info_out(fileInsideMod.getActualFileName()); | ||||
|  | ||||
|         if (!zipOutFile.open(QIODevice::WriteOnly, info_out)) | ||||
|         { | ||||
|         if (!zipOutFile.open(QIODevice::WriteOnly, info_out)) { | ||||
|             qCritical() << "Failed to open " << filename << " in the jar"; | ||||
|             fileInsideMod.close(); | ||||
|             return false; | ||||
|         } | ||||
|         if (!JlCompress::copyData(fileInsideMod, zipOutFile)) | ||||
|         { | ||||
|         if (!JlCompress::copyData(fileInsideMod, zipOutFile)) { | ||||
|             zipOutFile.close(); | ||||
|             fileInsideMod.close(); | ||||
|             qCritical() << "Failed to copy data of " << filename << " into the jar"; | ||||
| @@ -94,10 +89,11 @@ bool MMCZip::mergeZipFiles(QuaZip *into, QFileInfo from, QSet<QString> &containe | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| bool MMCZip::compressDirFiles(QuaZip *zip, QString dir, QFileInfoList files, bool followSymlinks) | ||||
| bool compressDirFiles(QuaZip* zip, QString dir, QFileInfoList files, bool followSymlinks) | ||||
| { | ||||
|     QDir directory(dir); | ||||
|     if (!directory.exists()) return false; | ||||
|     if (!directory.exists()) | ||||
|         return false; | ||||
|  | ||||
|     for (auto e : files) { | ||||
|         auto filePath = directory.relativeFilePath(e.absoluteFilePath()); | ||||
| @@ -109,17 +105,18 @@ bool MMCZip::compressDirFiles(QuaZip *zip, QString dir, QFileInfoList files, boo | ||||
|                 srcPath = e.canonicalFilePath(); | ||||
|             } | ||||
|         } | ||||
|         if( !JlCompress::compressFile(zip, srcPath, filePath)) return false; | ||||
|         if (!JlCompress::compressFile(zip, srcPath, filePath)) | ||||
|             return false; | ||||
|     } | ||||
|  | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| bool MMCZip::compressDirFiles(QString fileCompressed, QString dir, QFileInfoList files, bool followSymlinks) | ||||
| bool compressDirFiles(QString fileCompressed, QString dir, QFileInfoList files, bool followSymlinks) | ||||
| { | ||||
|     QuaZip zip(fileCompressed); | ||||
|     QDir().mkpath(QFileInfo(fileCompressed).absolutePath()); | ||||
|     if(!zip.open(QuaZip::mdCreate)) { | ||||
|     if (!zip.open(QuaZip::mdCreate)) { | ||||
|         QFile::remove(fileCompressed); | ||||
|         return false; | ||||
|     } | ||||
| @@ -127,7 +124,7 @@ bool MMCZip::compressDirFiles(QString fileCompressed, QString dir, QFileInfoList | ||||
|     auto result = compressDirFiles(&zip, dir, files, followSymlinks); | ||||
|  | ||||
|     zip.close(); | ||||
|     if(zip.getZipError()!=0) { | ||||
|     if (zip.getZipError() != 0) { | ||||
|         QFile::remove(fileCompressed); | ||||
|         return false; | ||||
|     } | ||||
| @@ -136,11 +133,10 @@ bool MMCZip::compressDirFiles(QString fileCompressed, QString dir, QFileInfoList | ||||
| } | ||||
|  | ||||
| // ours | ||||
| bool MMCZip::createModdedJar(QString sourceJarPath, QString targetJarPath, const QList<Mod*>& mods) | ||||
| bool createModdedJar(QString sourceJarPath, QString targetJarPath, const QList<Mod*>& mods) | ||||
| { | ||||
|     QuaZip zipOut(targetJarPath); | ||||
|     if (!zipOut.open(QuaZip::mdCreate)) | ||||
|     { | ||||
|     if (!zipOut.open(QuaZip::mdCreate)) { | ||||
|         QFile::remove(targetJarPath); | ||||
|         qCritical() << "Failed to open the minecraft.jar for modding"; | ||||
|         return false; | ||||
| @@ -151,37 +147,29 @@ bool MMCZip::createModdedJar(QString sourceJarPath, QString targetJarPath, const | ||||
|  | ||||
|     // Modify the jar | ||||
|     // This needs to be done in reverse-order to ensure we respect the loading order of components | ||||
|     for (auto i = mods.crbegin(); i != mods.crend(); i++) | ||||
|     { | ||||
|     for (auto i = mods.crbegin(); i != mods.crend(); i++) { | ||||
|         const auto* mod = *i; | ||||
|         // do not merge disabled mods. | ||||
|         if (!mod->enabled()) | ||||
|             continue; | ||||
|         if (mod->type() == ResourceType::ZIPFILE) | ||||
|         { | ||||
|             if (!mergeZipFiles(&zipOut, mod->fileinfo(), addedFiles)) | ||||
|             { | ||||
|         if (mod->type() == ResourceType::ZIPFILE) { | ||||
|             if (!mergeZipFiles(&zipOut, mod->fileinfo(), addedFiles)) { | ||||
|                 zipOut.close(); | ||||
|                 QFile::remove(targetJarPath); | ||||
|                 qCritical() << "Failed to add" << mod->fileinfo().fileName() << "to the jar."; | ||||
|                 return false; | ||||
|             } | ||||
|         } | ||||
|         else if (mod->type() == ResourceType::SINGLEFILE) | ||||
|         { | ||||
|         } else if (mod->type() == ResourceType::SINGLEFILE) { | ||||
|             // FIXME: buggy - does not work with addedFiles | ||||
|             auto filename = mod->fileinfo(); | ||||
|             if (!JlCompress::compressFile(&zipOut, filename.absoluteFilePath(), filename.fileName())) | ||||
|             { | ||||
|             if (!JlCompress::compressFile(&zipOut, filename.absoluteFilePath(), filename.fileName())) { | ||||
|                 zipOut.close(); | ||||
|                 QFile::remove(targetJarPath); | ||||
|                 qCritical() << "Failed to add" << mod->fileinfo().fileName() << "to the jar."; | ||||
|                 return false; | ||||
|             } | ||||
|             addedFiles.insert(filename.fileName()); | ||||
|         } | ||||
|         else if (mod->type() == ResourceType::FOLDER) | ||||
|         { | ||||
|         } else if (mod->type() == ResourceType::FOLDER) { | ||||
|             // untested, but seems to be unused / not possible to reach | ||||
|             // FIXME: buggy - does not work with addedFiles | ||||
|             auto filename = mod->fileinfo(); | ||||
| @@ -190,25 +178,21 @@ bool MMCZip::createModdedJar(QString sourceJarPath, QString targetJarPath, const | ||||
|             dir.cdUp(); | ||||
|             QString parent_dir = dir.absolutePath(); | ||||
|             auto files = QFileInfoList(); | ||||
|             MMCZip::collectFileListRecursively(what_to_zip, nullptr, &files, nullptr); | ||||
|             collectFileListRecursively(what_to_zip, nullptr, &files, nullptr); | ||||
|  | ||||
|             for (auto e : files) { | ||||
|                 if (addedFiles.contains(e.filePath())) | ||||
|                     files.removeAll(e); | ||||
|             } | ||||
|  | ||||
|             if (!MMCZip::compressDirFiles(&zipOut, parent_dir, files)) | ||||
|             { | ||||
|             if (!compressDirFiles(&zipOut, parent_dir, files)) { | ||||
|                 zipOut.close(); | ||||
|                 QFile::remove(targetJarPath); | ||||
|                 qCritical() << "Failed to add" << mod->fileinfo().fileName() << "to the jar."; | ||||
|                 return false; | ||||
|             } | ||||
|             qDebug() << "Adding folder " << filename.fileName() << " from " | ||||
|                      << filename.absoluteFilePath(); | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             qDebug() << "Adding folder " << filename.fileName() << " from " << filename.absoluteFilePath(); | ||||
|         } else { | ||||
|             // Make sure we do not continue launching when something is missing or undefined... | ||||
|             zipOut.close(); | ||||
|             QFile::remove(targetJarPath); | ||||
| @@ -217,8 +201,7 @@ bool MMCZip::createModdedJar(QString sourceJarPath, QString targetJarPath, const | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (!mergeZipFiles(&zipOut, QFileInfo(sourceJarPath), addedFiles, [](const QString key){return !key.contains("META-INF");})) | ||||
|     { | ||||
|     if (!mergeZipFiles(&zipOut, QFileInfo(sourceJarPath), addedFiles, [](const QString key) { return !key.contains("META-INF"); })) { | ||||
|         zipOut.close(); | ||||
|         QFile::remove(targetJarPath); | ||||
|         qCritical() << "Failed to insert minecraft.jar contents."; | ||||
| @@ -227,8 +210,7 @@ bool MMCZip::createModdedJar(QString sourceJarPath, QString targetJarPath, const | ||||
|  | ||||
|     // Recompress the jar | ||||
|     zipOut.close(); | ||||
|     if (zipOut.getZipError() != 0) | ||||
|     { | ||||
|     if (zipOut.getZipError() != 0) { | ||||
|         QFile::remove(targetJarPath); | ||||
|         qCritical() << "Failed to finalize minecraft.jar!"; | ||||
|         return false; | ||||
| @@ -237,7 +219,7 @@ bool MMCZip::createModdedJar(QString sourceJarPath, QString targetJarPath, const | ||||
| } | ||||
|  | ||||
| // ours | ||||
| QString MMCZip::findFolderOfFileInZip(QuaZip* zip, const QString& what, const QStringList& ignore_paths, const QString& root) | ||||
| QString findFolderOfFileInZip(QuaZip* zip, const QString& what, const QStringList& ignore_paths, const QString& root) | ||||
| { | ||||
|     QuaZipDir rootDir(zip, root); | ||||
|     for (auto&& fileName : rootDir.entryList(QDir::Files)) { | ||||
| @@ -261,27 +243,23 @@ QString MMCZip::findFolderOfFileInZip(QuaZip* zip, const QString& what, const QS | ||||
| } | ||||
|  | ||||
| // ours | ||||
| bool MMCZip::findFilesInZip(QuaZip * zip, const QString & what, QStringList & result, const QString &root) | ||||
| bool findFilesInZip(QuaZip* zip, const QString& what, QStringList& result, const QString& root) | ||||
| { | ||||
|     QuaZipDir rootDir(zip, root); | ||||
|     for(auto fileName: rootDir.entryList(QDir::Files)) | ||||
|     { | ||||
|         if(fileName == what) | ||||
|         { | ||||
|     for (auto fileName : rootDir.entryList(QDir::Files)) { | ||||
|         if (fileName == what) { | ||||
|             result.append(root); | ||||
|             return true; | ||||
|         } | ||||
|     } | ||||
|     for(auto fileName: rootDir.entryList(QDir::Dirs)) | ||||
|     { | ||||
|     for (auto fileName : rootDir.entryList(QDir::Dirs)) { | ||||
|         findFilesInZip(zip, what, result, root + fileName); | ||||
|     } | ||||
|     return !result.isEmpty(); | ||||
| } | ||||
|  | ||||
|  | ||||
| // ours | ||||
| std::optional<QStringList> MMCZip::extractSubDir(QuaZip *zip, const QString & subdir, const QString &target) | ||||
| std::optional<QStringList> extractSubDir(QuaZip* zip, const QString& subdir, const QString& target) | ||||
| { | ||||
|     auto target_top_dir = QUrl::fromLocalFile(target); | ||||
|  | ||||
| @@ -289,16 +267,13 @@ std::optional<QStringList> MMCZip::extractSubDir(QuaZip *zip, const QString & su | ||||
|  | ||||
|     qDebug() << "Extracting subdir" << subdir << "from" << zip->getZipName() << "to" << target; | ||||
|     auto numEntries = zip->getEntriesCount(); | ||||
|     if(numEntries < 0) { | ||||
|     if (numEntries < 0) { | ||||
|         qWarning() << "Failed to enumerate files in archive"; | ||||
|         return std::nullopt; | ||||
|     } | ||||
|     else if(numEntries == 0) { | ||||
|     } else if (numEntries == 0) { | ||||
|         qDebug() << "Extracting empty archives seems odd..."; | ||||
|         return extracted; | ||||
|     } | ||||
|     else if (!zip->goToFirstFile()) | ||||
|     { | ||||
|     } else if (!zip->goToFirstFile()) { | ||||
|         qWarning() << "Failed to seek to first file in zip"; | ||||
|         return std::nullopt; | ||||
|     } | ||||
| @@ -334,7 +309,8 @@ std::optional<QStringList> MMCZip::extractSubDir(QuaZip *zip, const QString & su | ||||
|         } | ||||
|  | ||||
|         if (!target_top_dir.isParentOf(QUrl::fromLocalFile(target_file_path))) { | ||||
|             qWarning() << "Extracting" << relative_file_name << "was cancelled, because it was effectively outside of the target path" << target; | ||||
|             qWarning() << "Extracting" << relative_file_name << "was cancelled, because it was effectively outside of the target path" | ||||
|                        << target; | ||||
|             return std::nullopt; | ||||
|         } | ||||
|  | ||||
| @@ -345,7 +321,8 @@ std::optional<QStringList> MMCZip::extractSubDir(QuaZip *zip, const QString & su | ||||
|         } | ||||
|  | ||||
|         extracted.append(target_file_path); | ||||
|         QFile::setPermissions(target_file_path, QFileDevice::Permission::ReadUser | QFileDevice::Permission::WriteUser | QFileDevice::Permission::ExeUser); | ||||
|         QFile::setPermissions(target_file_path, | ||||
|                               QFileDevice::Permission::ReadUser | QFileDevice::Permission::WriteUser | QFileDevice::Permission::ExeUser); | ||||
|  | ||||
|         qDebug() << "Extracted file" << relative_file_name << "to" << target_file_path; | ||||
|     } while (zip->goToNextFile()); | ||||
| @@ -354,66 +331,66 @@ std::optional<QStringList> MMCZip::extractSubDir(QuaZip *zip, const QString & su | ||||
| } | ||||
|  | ||||
| // ours | ||||
| bool MMCZip::extractRelFile(QuaZip *zip, const QString &file, const QString &target) | ||||
| bool extractRelFile(QuaZip* zip, const QString& file, const QString& target) | ||||
| { | ||||
|     return JlCompress::extractFile(zip, file, target); | ||||
| } | ||||
|  | ||||
| // ours | ||||
| std::optional<QStringList> MMCZip::extractDir(QString fileCompressed, QString dir) | ||||
| std::optional<QStringList> extractDir(QString fileCompressed, QString dir) | ||||
| { | ||||
|     QuaZip zip(fileCompressed); | ||||
|     if (!zip.open(QuaZip::mdUnzip)) | ||||
|     { | ||||
|     if (!zip.open(QuaZip::mdUnzip)) { | ||||
|         // check if this is a minimum size empty zip file... | ||||
|         QFileInfo fileInfo(fileCompressed); | ||||
|         if(fileInfo.size() == 22) { | ||||
|         if (fileInfo.size() == 22) { | ||||
|             return QStringList(); | ||||
|         } | ||||
|         qWarning() << "Could not open archive for unzipping:" << fileCompressed << "Error:" << zip.getZipError();; | ||||
|         qWarning() << "Could not open archive for unzipping:" << fileCompressed << "Error:" << zip.getZipError(); | ||||
|         ; | ||||
|         return std::nullopt; | ||||
|     } | ||||
|     return MMCZip::extractSubDir(&zip, "", dir); | ||||
|     return extractSubDir(&zip, "", dir); | ||||
| } | ||||
|  | ||||
| // ours | ||||
| std::optional<QStringList> MMCZip::extractDir(QString fileCompressed, QString subdir, QString dir) | ||||
| std::optional<QStringList> extractDir(QString fileCompressed, QString subdir, QString dir) | ||||
| { | ||||
|     QuaZip zip(fileCompressed); | ||||
|     if (!zip.open(QuaZip::mdUnzip)) | ||||
|     { | ||||
|     if (!zip.open(QuaZip::mdUnzip)) { | ||||
|         // check if this is a minimum size empty zip file... | ||||
|         QFileInfo fileInfo(fileCompressed); | ||||
|         if(fileInfo.size() == 22) { | ||||
|         if (fileInfo.size() == 22) { | ||||
|             return QStringList(); | ||||
|         } | ||||
|         qWarning() << "Could not open archive for unzipping:" << fileCompressed << "Error:" << zip.getZipError();; | ||||
|         qWarning() << "Could not open archive for unzipping:" << fileCompressed << "Error:" << zip.getZipError(); | ||||
|         ; | ||||
|         return std::nullopt; | ||||
|     } | ||||
|     return MMCZip::extractSubDir(&zip, subdir, dir); | ||||
|     return extractSubDir(&zip, subdir, dir); | ||||
| } | ||||
|  | ||||
| // ours | ||||
| bool MMCZip::extractFile(QString fileCompressed, QString file, QString target) | ||||
| bool extractFile(QString fileCompressed, QString file, QString target) | ||||
| { | ||||
|     QuaZip zip(fileCompressed); | ||||
|     if (!zip.open(QuaZip::mdUnzip)) | ||||
|     { | ||||
|     if (!zip.open(QuaZip::mdUnzip)) { | ||||
|         // check if this is a minimum size empty zip file... | ||||
|         QFileInfo fileInfo(fileCompressed); | ||||
|         if(fileInfo.size() == 22) { | ||||
|         if (fileInfo.size() == 22) { | ||||
|             return true; | ||||
|         } | ||||
|         qWarning() << "Could not open archive for unzipping:" << fileCompressed << "Error:" << zip.getZipError(); | ||||
|         return false; | ||||
|     } | ||||
|     return MMCZip::extractRelFile(&zip, file, target); | ||||
|     return extractRelFile(&zip, file, target); | ||||
| } | ||||
|  | ||||
| bool MMCZip::collectFileListRecursively(const QString& rootDir, const QString& subDir, QFileInfoList *files, | ||||
|                                         MMCZip::FilterFunction excludeFilter) { | ||||
| bool collectFileListRecursively(const QString& rootDir, const QString& subDir, QFileInfoList* files, FilterFunction excludeFilter) | ||||
| { | ||||
|     QDir rootDirectory(rootDir); | ||||
|     if (!rootDirectory.exists()) return false; | ||||
|     if (!rootDirectory.exists()) | ||||
|         return false; | ||||
|  | ||||
|     QDir directory; | ||||
|     if (subDir == nullptr) | ||||
| @@ -421,25 +398,107 @@ bool MMCZip::collectFileListRecursively(const QString& rootDir, const QString& s | ||||
|     else | ||||
|         directory = QDir(subDir); | ||||
|  | ||||
|     if (!directory.exists()) return false;  // shouldn't ever happen | ||||
|     if (!directory.exists()) | ||||
|         return false;  // shouldn't ever happen | ||||
|  | ||||
|     // recurse directories | ||||
|     QFileInfoList entries = directory.entryInfoList(QDir::AllDirs | QDir::NoDotAndDotDot | QDir::Hidden); | ||||
|     for (const auto& e: entries) { | ||||
|     for (const auto& e : entries) { | ||||
|         if (!collectFileListRecursively(rootDir, e.filePath(), files, excludeFilter)) | ||||
|             return false; | ||||
|     } | ||||
|  | ||||
|     // collect files | ||||
|     entries = directory.entryInfoList(QDir::Files); | ||||
|     for (const auto& e: entries) { | ||||
|     for (const auto& e : entries) { | ||||
|         QString relativeFilePath = rootDirectory.relativeFilePath(e.absoluteFilePath()); | ||||
|         if (excludeFilter && excludeFilter(relativeFilePath)) { | ||||
|             qDebug() << "Skipping file " << relativeFilePath; | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         files->append(e);  // we want the original paths for MMCZip::compressDirFiles | ||||
|         files->append(e);  // we want the original paths for compressDirFiles | ||||
|     } | ||||
|     return true; | ||||
| } | ||||
|  | ||||
| void ExportToZipTask::executeTask() | ||||
| { | ||||
|     setStatus("Adding files..."); | ||||
|     setProgress(0, m_files.length()); | ||||
|     m_build_zip_future = QtConcurrent::run(QThreadPool::globalInstance(), [this]() { return exportZip(); }); | ||||
|     connect(&m_build_zip_watcher, &QFutureWatcher<ZipResult>::finished, this, &ExportToZipTask::finish); | ||||
|     m_build_zip_watcher.setFuture(m_build_zip_future); | ||||
| } | ||||
|  | ||||
| auto ExportToZipTask::exportZip() -> ZipResult | ||||
| { | ||||
|     if (!m_dir.exists()) { | ||||
|         return ZipResult(tr("Folder doesn't exist")); | ||||
|     } | ||||
|     if (!m_output.isOpen() && !m_output.open(QuaZip::mdCreate)) { | ||||
|         return ZipResult(tr("Could not create file")); | ||||
|     } | ||||
|  | ||||
|     for (auto fileName : m_extra_files.keys()) { | ||||
|         if (m_build_zip_future.isCanceled()) | ||||
|             return ZipResult(); | ||||
|         QuaZipFile indexFile(&m_output); | ||||
|         if (!indexFile.open(QIODevice::WriteOnly, QuaZipNewInfo(fileName))) { | ||||
|             return ZipResult(tr("Could not create:") + fileName); | ||||
|         } | ||||
|         indexFile.write(m_extra_files[fileName]); | ||||
|     } | ||||
|  | ||||
|     for (const QFileInfo& file : m_files) { | ||||
|         if (m_build_zip_future.isCanceled()) | ||||
|             return ZipResult(); | ||||
|  | ||||
|         auto absolute = file.absoluteFilePath(); | ||||
|         auto relative = m_dir.relativeFilePath(absolute); | ||||
|         setStatus("Compresing: " + relative); | ||||
|         setProgress(m_progress + 1, m_progressTotal); | ||||
|         if (m_follow_symlinks) { | ||||
|             if (file.isSymLink()) | ||||
|                 absolute = file.symLinkTarget(); | ||||
|             else | ||||
|                 absolute = file.canonicalFilePath(); | ||||
|         } | ||||
|  | ||||
|         if (!m_exclude_files.contains(relative) && !JlCompress::compressFile(&m_output, absolute, m_destination_prefix + relative)) { | ||||
|             return ZipResult(tr("Could not read and compress %1").arg(relative)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     m_output.close(); | ||||
|     if (m_output.getZipError() != 0) { | ||||
|         return ZipResult(tr("A zip error occurred")); | ||||
|     } | ||||
|     return ZipResult(); | ||||
| } | ||||
|  | ||||
| void ExportToZipTask::finish() | ||||
| { | ||||
|     if (m_build_zip_future.isCanceled()) { | ||||
|         QFile::remove(m_output_path); | ||||
|         emitAborted(); | ||||
|     } else if (auto result = m_build_zip_future.result(); result.has_value()) { | ||||
|         QFile::remove(m_output_path); | ||||
|         emitFailed(result.value()); | ||||
|     } else { | ||||
|         emitSucceeded(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| bool ExportToZipTask::abort() | ||||
| { | ||||
|     if (m_build_zip_future.isRunning()) { | ||||
|         m_build_zip_future.cancel(); | ||||
|         // NOTE: Here we don't do `emitAborted()` because it will be done when `m_build_zip_future` actually cancels, which may not occur | ||||
|         // immediately. | ||||
|         return true; | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| }  // namespace MMCZip | ||||
| @@ -1,7 +1,8 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  PolyMC - Minecraft Launcher | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net> | ||||
|  *  Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
| @@ -35,26 +36,30 @@ | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include <QString> | ||||
| #include <QFileInfo> | ||||
| #include <QSet> | ||||
| #include "minecraft/mod/Mod.h" | ||||
| #include <functional> | ||||
|  | ||||
| #include <quazip.h> | ||||
| #include <quazip/JlCompress.h> | ||||
| #include <QDir> | ||||
| #include <QFileInfo> | ||||
| #include <QFuture> | ||||
| #include <QFutureWatcher> | ||||
| #include <QHash> | ||||
| #include <QSet> | ||||
| #include <QString> | ||||
| #include <functional> | ||||
| #include <memory> | ||||
| #include <optional> | ||||
| #include "minecraft/mod/Mod.h" | ||||
| #include "tasks/Task.h" | ||||
|  | ||||
| namespace MMCZip | ||||
| { | ||||
|     using FilterFunction = std::function<bool(const QString &)>; | ||||
| namespace MMCZip { | ||||
| using FilterFunction = std::function<bool(const QString&)>; | ||||
|  | ||||
|     /** | ||||
| /** | ||||
|  * Merge two zip files, using a filter function | ||||
|  */ | ||||
|     bool mergeZipFiles(QuaZip *into, QFileInfo from, QSet<QString> &contained, | ||||
|                                             const FilterFunction filter = nullptr); | ||||
| bool mergeZipFiles(QuaZip* into, QFileInfo from, QSet<QString>& contained, const FilterFunction filter = nullptr); | ||||
|  | ||||
|     /** | ||||
| /** | ||||
|  * Compress directory, by providing a list of files to compress | ||||
|  * \param zip target archive | ||||
|  * \param dir directory that will be compressed (to compress with relative paths) | ||||
| @@ -62,9 +67,9 @@ namespace MMCZip | ||||
|  * \param followSymlinks should follow symlinks when compressing file data | ||||
|  * \return true for success or false for failure | ||||
|  */ | ||||
|     bool compressDirFiles(QuaZip *zip, QString dir, QFileInfoList files, bool followSymlinks = false); | ||||
| bool compressDirFiles(QuaZip* zip, QString dir, QFileInfoList files, bool followSymlinks = false); | ||||
|  | ||||
|     /** | ||||
| /** | ||||
|  * Compress directory, by providing a list of files to compress | ||||
|  * \param fileCompressed target archive file | ||||
|  * \param dir directory that will be compressed (to compress with relative paths) | ||||
| @@ -72,47 +77,47 @@ namespace MMCZip | ||||
|  * \param followSymlinks should follow symlinks when compressing file data | ||||
|  * \return true for success or false for failure | ||||
|  */ | ||||
|     bool compressDirFiles(QString fileCompressed, QString dir, QFileInfoList files, bool followSymlinks = false); | ||||
| bool compressDirFiles(QString fileCompressed, QString dir, QFileInfoList files, bool followSymlinks = false); | ||||
|  | ||||
|     /** | ||||
| /** | ||||
|  * take a source jar, add mods to it, resulting in target jar | ||||
|  */ | ||||
|     bool createModdedJar(QString sourceJarPath, QString targetJarPath, const QList<Mod*>& mods); | ||||
| bool createModdedJar(QString sourceJarPath, QString targetJarPath, const QList<Mod*>& mods); | ||||
|  | ||||
|     /** | ||||
| /** | ||||
|  * Find a single file in archive by file name (not path) | ||||
|  * | ||||
|  * \param ignore_paths paths to skip when recursing the search | ||||
|  * | ||||
|  * \return the path prefix where the file is | ||||
|  */ | ||||
|     QString findFolderOfFileInZip(QuaZip * zip, const QString & what, const QStringList& ignore_paths = {}, const QString &root = QString("")); | ||||
| QString findFolderOfFileInZip(QuaZip* zip, const QString& what, const QStringList& ignore_paths = {}, const QString& root = QString("")); | ||||
|  | ||||
|     /** | ||||
| /** | ||||
|  * Find a multiple files of the same name in archive by file name | ||||
|  * If a file is found in a path, no deeper paths are searched | ||||
|  * | ||||
|  * \return true if anything was found | ||||
|  */ | ||||
|     bool findFilesInZip(QuaZip * zip, const QString & what, QStringList & result, const QString &root = QString()); | ||||
| bool findFilesInZip(QuaZip* zip, const QString& what, QStringList& result, const QString& root = QString()); | ||||
|  | ||||
|     /** | ||||
| /** | ||||
|  * Extract a subdirectory from an archive | ||||
|  */ | ||||
|     std::optional<QStringList> extractSubDir(QuaZip *zip, const QString & subdir, const QString &target); | ||||
| std::optional<QStringList> extractSubDir(QuaZip* zip, const QString& subdir, const QString& target); | ||||
|  | ||||
|     bool extractRelFile(QuaZip *zip, const QString & file, const QString &target); | ||||
| bool extractRelFile(QuaZip* zip, const QString& file, const QString& target); | ||||
|  | ||||
|     /** | ||||
| /** | ||||
|  * Extract a whole archive. | ||||
|  * | ||||
|  * \param fileCompressed The name of the archive. | ||||
|  * \param dir The directory to extract to, the current directory if left empty. | ||||
|  * \return The list of the full paths of the files extracted, empty on failure. | ||||
|  */ | ||||
|     std::optional<QStringList> extractDir(QString fileCompressed, QString dir); | ||||
| std::optional<QStringList> extractDir(QString fileCompressed, QString dir); | ||||
|  | ||||
|     /** | ||||
| /** | ||||
|  * Extract a subdirectory from an archive | ||||
|  * | ||||
|  * \param fileCompressed The name of the archive. | ||||
| @@ -120,9 +125,9 @@ namespace MMCZip | ||||
|  * \param dir The directory to extract to, the current directory if left empty. | ||||
|  * \return The list of the full paths of the files extracted, empty on failure. | ||||
|  */ | ||||
|     std::optional<QStringList> extractDir(QString fileCompressed, QString subdir, QString dir); | ||||
| std::optional<QStringList> extractDir(QString fileCompressed, QString subdir, QString dir); | ||||
|  | ||||
|     /** | ||||
| /** | ||||
|  * Extract a single file from an archive into a directory | ||||
|  * | ||||
|  * \param fileCompressed The name of the archive. | ||||
| @@ -130,9 +135,9 @@ namespace MMCZip | ||||
|  * \param dir The directory to extract to, the current directory if left empty. | ||||
|  * \return true for success or false for failure | ||||
|  */ | ||||
|     bool extractFile(QString fileCompressed, QString file, QString dir); | ||||
| bool extractFile(QString fileCompressed, QString file, QString dir); | ||||
|  | ||||
|     /** | ||||
| /** | ||||
|  * Populate a QFileInfoList with a directory tree recursively, while allowing to excludeFilter what shouldn't be included. | ||||
|  * \param rootDir directory to start off | ||||
|  * \param subDir subdirectory, should be nullptr for first invocation | ||||
| @@ -140,5 +145,48 @@ namespace MMCZip | ||||
|  * \param excludeFilter function to excludeFilter which files shouldn't be included (returning true means to excude) | ||||
|  * \return true for success or false for failure | ||||
|  */ | ||||
|     bool collectFileListRecursively(const QString &rootDir, const QString &subDir, QFileInfoList *files, FilterFunction excludeFilter); | ||||
| } | ||||
| bool collectFileListRecursively(const QString& rootDir, const QString& subDir, QFileInfoList* files, FilterFunction excludeFilter); | ||||
|  | ||||
| class ExportToZipTask : public Task { | ||||
|    public: | ||||
|     ExportToZipTask(QString outputPath, QDir dir, QFileInfoList files, QString destinationPrefix = "", bool followSymlinks = false) | ||||
|         : m_output_path(outputPath) | ||||
|         , m_output(outputPath) | ||||
|         , m_dir(dir) | ||||
|         , m_files(files) | ||||
|         , m_destination_prefix(destinationPrefix) | ||||
|         , m_follow_symlinks(followSymlinks) | ||||
|     { | ||||
|         setAbortable(true); | ||||
|     }; | ||||
|     ExportToZipTask(QString outputPath, QString dir, QFileInfoList files, QString destinationPrefix = "", bool followSymlinks = false) | ||||
|         : ExportToZipTask(outputPath, QDir(dir), files, destinationPrefix, followSymlinks){}; | ||||
|  | ||||
|     virtual ~ExportToZipTask() = default; | ||||
|  | ||||
|     void setExcludeFiles(QStringList excludeFiles) { m_exclude_files = excludeFiles; } | ||||
|     void addExtraFile(QString fileName, QByteArray data) { m_extra_files.insert(fileName, data); } | ||||
|  | ||||
|     typedef std::optional<QString> ZipResult; | ||||
|  | ||||
|    protected: | ||||
|     virtual void executeTask() override; | ||||
|     bool abort() override; | ||||
|  | ||||
|     ZipResult exportZip(); | ||||
|     void finish(); | ||||
|  | ||||
|    private: | ||||
|     QString m_output_path; | ||||
|     QuaZip m_output; | ||||
|     QDir m_dir; | ||||
|     QFileInfoList m_files; | ||||
|     QString m_destination_prefix; | ||||
|     bool m_follow_symlinks; | ||||
|     QStringList m_exclude_files; | ||||
|     QHash<QString, QByteArray> m_extra_files; | ||||
|  | ||||
|     QFuture<ZipResult> m_build_zip_future; | ||||
|     QFutureWatcher<ZipResult> m_build_zip_watcher; | ||||
| }; | ||||
| }  // namespace MMCZip | ||||
|   | ||||
| @@ -1,29 +1,64 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, version 3. | ||||
|  * | ||||
|  *  This program is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
|  | ||||
| #include "JavaInstall.h" | ||||
|  | ||||
| #include "BaseVersion.h" | ||||
| #include "StringUtils.h" | ||||
|  | ||||
| bool JavaInstall::operator<(const JavaInstall &rhs) | ||||
| bool JavaInstall::operator<(const JavaInstall& rhs) | ||||
| { | ||||
|     auto archCompare = StringUtils::naturalCompare(arch, rhs.arch, Qt::CaseInsensitive); | ||||
|     if(archCompare != 0) | ||||
|     if (archCompare != 0) | ||||
|         return archCompare < 0; | ||||
|     if(id < rhs.id) | ||||
|     { | ||||
|     if (id < rhs.id) { | ||||
|         return true; | ||||
|     } | ||||
|     if(id > rhs.id) | ||||
|     { | ||||
|     if (id > rhs.id) { | ||||
|         return false; | ||||
|     } | ||||
|     return StringUtils::naturalCompare(path, rhs.path, Qt::CaseInsensitive) < 0; | ||||
| } | ||||
|  | ||||
| bool JavaInstall::operator==(const JavaInstall &rhs) | ||||
| bool JavaInstall::operator==(const JavaInstall& rhs) | ||||
| { | ||||
|     return arch == rhs.arch && id == rhs.id && path == rhs.path; | ||||
| } | ||||
|  | ||||
| bool JavaInstall::operator>(const JavaInstall &rhs) | ||||
| bool JavaInstall::operator>(const JavaInstall& rhs) | ||||
| { | ||||
|     return (!operator<(rhs)) && (!operator==(rhs)); | ||||
| } | ||||
|  | ||||
| bool JavaInstall::operator<(BaseVersion& a) | ||||
| { | ||||
|     try { | ||||
|         return operator<(dynamic_cast<JavaInstall&>(a)); | ||||
|     } catch (const std::bad_cast& e) { | ||||
|         return BaseVersion::operator<(a); | ||||
|     } | ||||
| } | ||||
|  | ||||
| bool JavaInstall::operator>(BaseVersion& a) | ||||
| { | ||||
|     try { | ||||
|         return operator>(dynamic_cast<JavaInstall&>(a)); | ||||
|     } catch (const std::bad_cast& e) { | ||||
|         return BaseVersion::operator>(a); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,33 +1,40 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, version 3. | ||||
|  * | ||||
|  *  This program is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include "BaseVersion.h" | ||||
| #include "JavaVersion.h" | ||||
|  | ||||
| struct JavaInstall : public BaseVersion | ||||
| { | ||||
|     JavaInstall(){} | ||||
|     JavaInstall(QString id, QString arch, QString path) | ||||
|     : id(id), arch(arch), path(path) | ||||
|     { | ||||
|     } | ||||
|     virtual QString descriptor() | ||||
|     { | ||||
|         return id.toString(); | ||||
|     } | ||||
| struct JavaInstall : public BaseVersion { | ||||
|     JavaInstall() {} | ||||
|     JavaInstall(QString id, QString arch, QString path) : id(id), arch(arch), path(path) {} | ||||
|     virtual QString descriptor() { return id.toString(); } | ||||
|  | ||||
|     virtual QString name() | ||||
|     { | ||||
|         return id.toString(); | ||||
|     } | ||||
|     virtual QString name() { return id.toString(); } | ||||
|  | ||||
|     virtual QString typeString() const | ||||
|     { | ||||
|         return arch; | ||||
|     } | ||||
|     virtual QString typeString() const { return arch; } | ||||
|  | ||||
|     bool operator<(const JavaInstall & rhs); | ||||
|     bool operator==(const JavaInstall & rhs); | ||||
|     bool operator>(const JavaInstall & rhs); | ||||
|     virtual bool operator<(BaseVersion& a) override; | ||||
|     virtual bool operator>(BaseVersion& a) override; | ||||
|     bool operator<(const JavaInstall& rhs); | ||||
|     bool operator==(const JavaInstall& rhs); | ||||
|     bool operator>(const JavaInstall& rhs); | ||||
|  | ||||
|     JavaVersion id; | ||||
|     QString arch; | ||||
|   | ||||
| @@ -4,6 +4,7 @@ | ||||
|  *  Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net> | ||||
|  *  Copyright (C) 2022 Jamie Mansfield <jmansfield@cadixdev.org> | ||||
|  *  Copyright (C) 2022 TheKodeToad <TheKodeToad@proton.me> | ||||
|  *  Copyright (c) 2023 seth <getchoo at tuta dot io> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
| @@ -186,6 +187,10 @@ void MinecraftInstance::loadSpecificSettings() | ||||
|         m_settings->registerOverride(global_settings->getSetting("CloseAfterLaunch"), miscellaneousOverride); | ||||
|         m_settings->registerOverride(global_settings->getSetting("QuitAfterGameStop"), miscellaneousOverride); | ||||
|  | ||||
|         // Mod loader specific options | ||||
|         auto modLoaderSettings = m_settings->registerSetting("OverrideModLoaderSettings", false); | ||||
|         m_settings->registerOverride(global_settings->getSetting("DisableQuiltBeacon"), modLoaderSettings); | ||||
|  | ||||
|         m_settings->set("InstanceType", "OneSix"); | ||||
|     } | ||||
|  | ||||
| @@ -391,6 +396,12 @@ QStringList MinecraftInstance::extraArguments() | ||||
|         agent->library()->getApplicableFiles(runtimeContext(), jar, temp1, temp2, temp3, getLocalLibraryPath()); | ||||
|         list.append("-javaagent:"+jar[0]+(agent->argument().isEmpty() ? "" : "="+agent->argument())); | ||||
|     } | ||||
|  | ||||
|     { | ||||
|         const auto loaders = version->getModLoaders(); | ||||
|         if (loaders.has_value() && loaders.value() & ResourceAPI::Quilt && settings()->get("DisableQuiltBeacon").toBool()) | ||||
|             list.append("-Dloader.disable_beacon=true"); | ||||
|     } | ||||
|     return list; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -44,7 +44,11 @@ static const QMap<PackedResourceType, QString> s_packed_type_names = { | ||||
| namespace ResourceUtils { | ||||
| PackedResourceType identify(QFileInfo file){ | ||||
|     if (file.exists() && file.isFile()) { | ||||
|         if (ResourcePackUtils::validate(file)) { | ||||
|         if (ModUtils::validate(file)) { | ||||
|             // mods can contain resource and data packs so they must be tested first | ||||
|             qDebug() << file.fileName() << "is a mod"; | ||||
|             return PackedResourceType::Mod; | ||||
|         } else if (ResourcePackUtils::validate(file)) { | ||||
|             qDebug() << file.fileName() << "is a resource pack"; | ||||
|             return PackedResourceType::ResourcePack; | ||||
|         } else if (TexturePackUtils::validate(file)) { | ||||
| @@ -53,9 +57,6 @@ PackedResourceType identify(QFileInfo file){ | ||||
|         } else if (DataPackUtils::validate(file)) { | ||||
|             qDebug() << file.fileName() << "is a data pack"; | ||||
|             return PackedResourceType::DataPack; | ||||
|         } else if (ModUtils::validate(file)) { | ||||
|             qDebug() << file.fileName() << "is a mod"; | ||||
|             return PackedResourceType::Mod; | ||||
|         } else if (WorldSaveUtils::validate(file)) { | ||||
|             qDebug() << file.fileName() << "is a world save"; | ||||
|             return PackedResourceType::WorldSave; | ||||
|   | ||||
| @@ -145,7 +145,8 @@ void EnsureMetadataTask::executeTask() | ||||
|         connect(project_task.get(), &Task::finished, this, [=] { | ||||
|             invalidade_leftover(); | ||||
|             project_task->deleteLater(); | ||||
|             m_current_task = nullptr; | ||||
|             if (m_current_task) | ||||
|                 m_current_task.reset(); | ||||
|         }); | ||||
|  | ||||
|         m_current_task = project_task; | ||||
| @@ -154,7 +155,8 @@ void EnsureMetadataTask::executeTask() | ||||
|  | ||||
|     connect(version_task.get(), &Task::finished, [=] { | ||||
|         version_task->deleteLater(); | ||||
|         m_current_task = nullptr; | ||||
|         if (m_current_task) | ||||
|             m_current_task.reset(); | ||||
|     }); | ||||
|  | ||||
|     if (m_mods.size() > 1) | ||||
|   | ||||
| @@ -21,6 +21,10 @@ bool Flame::FileResolvingTask::abort() | ||||
|  | ||||
| void Flame::FileResolvingTask::executeTask() | ||||
| { | ||||
|     if (m_toProcess.files.isEmpty()) {  // no file to resolve so leave it empty and emit success immediately | ||||
|         emitSucceeded(); | ||||
|         return; | ||||
|     } | ||||
|     setStatus(tr("Resolving mod IDs...")); | ||||
|     setProgress(0, 3); | ||||
|     m_dljob.reset(new NetJob("Mod id resolver", m_network)); | ||||
| @@ -128,12 +132,13 @@ void Flame::FileResolvingTask::netJobFinished() | ||||
|     m_checkJob->start(); | ||||
| } | ||||
|  | ||||
| void Flame::FileResolvingTask::modrinthCheckFinished() { | ||||
| void Flame::FileResolvingTask::modrinthCheckFinished() | ||||
| { | ||||
|     setProgress(2, 3); | ||||
|     qDebug() << "Finished with blocked mods : " << blockedProjects.size(); | ||||
|  | ||||
|     for (auto it = blockedProjects.keyBegin(); it != blockedProjects.keyEnd(); it++) { | ||||
|         auto &out = *it; | ||||
|         auto& out = *it; | ||||
|         auto bytes = blockedProjects[out]; | ||||
|         if (!out->resolved) { | ||||
|             continue; | ||||
| @@ -153,15 +158,13 @@ void Flame::FileResolvingTask::modrinthCheckFinished() { | ||||
|             out->resolved = false; | ||||
|         } | ||||
|     } | ||||
|     //copy to an output list and filter out projects found on modrinth | ||||
|     // copy to an output list and filter out projects found on modrinth | ||||
|     auto block = std::make_shared<QList<File*>>(); | ||||
|     auto it = blockedProjects.keys(); | ||||
|     std::copy_if(it.begin(), it.end(), std::back_inserter(*block), [](File *f) { | ||||
|         return !f->resolved; | ||||
|     }); | ||||
|     //Display not found mods early | ||||
|     std::copy_if(it.begin(), it.end(), std::back_inserter(*block), [](File* f) { return !f->resolved; }); | ||||
|     // Display not found mods early | ||||
|     if (!block->empty()) { | ||||
|         //blocked mods found, we need the slug for displaying.... we need another job :D ! | ||||
|         // blocked mods found, we need the slug for displaying.... we need another job :D ! | ||||
|         m_slugJob.reset(new NetJob("Slug Job", m_network)); | ||||
|         int index = 0; | ||||
|         for (auto mod : *block) { | ||||
| @@ -173,8 +176,8 @@ void Flame::FileResolvingTask::modrinthCheckFinished() { | ||||
|             QObject::connect(dl.get(), &Net::Download::succeeded, [block, index, output]() { | ||||
|                 auto mod = block->at(index);  // use the shared_ptr so it is captured and only freed when we are done | ||||
|                 auto json = QJsonDocument::fromJson(*output); | ||||
|                 auto base = Json::requireString(Json::requireObject(Json::requireObject(Json::requireObject(json),"data"),"links"), | ||||
|                         "websiteUrl"); | ||||
|                 auto base = | ||||
|                     Json::requireString(Json::requireObject(Json::requireObject(Json::requireObject(json), "data"), "links"), "websiteUrl"); | ||||
|                 auto link = QString("%1/download/%2").arg(base, QString::number(mod->fileId)); | ||||
|                 mod->websiteUrl = link; | ||||
|             }); | ||||
|   | ||||
| @@ -563,6 +563,8 @@ void FlameCreationTask::validateZIPResouces() | ||||
|                 if (FS::move(localPath, destPath)) { | ||||
|                     return destPath; | ||||
|                 } | ||||
|             } else { | ||||
|                 qDebug() << "Target folder of" << fileName << "is correct at" << targetFolder; | ||||
|             } | ||||
|             return localPath; | ||||
|         }; | ||||
| @@ -584,6 +586,9 @@ void FlameCreationTask::validateZIPResouces() | ||||
|         QString worldPath; | ||||
|  | ||||
|         switch (type) { | ||||
|             case PackedResourceType::Mod : | ||||
|                 validatePath(fileName, targetFolder, "mods"); | ||||
|                 break; | ||||
|             case PackedResourceType::ResourcePack : | ||||
|                 validatePath(fileName, targetFolder, "resourcepacks"); | ||||
|                 break; | ||||
| @@ -593,9 +598,6 @@ void FlameCreationTask::validateZIPResouces() | ||||
|             case PackedResourceType::DataPack : | ||||
|                 validatePath(fileName, targetFolder, "datapacks"); | ||||
|                 break; | ||||
|             case PackedResourceType::Mod : | ||||
|                 validatePath(fileName, targetFolder, "mods"); | ||||
|                 break; | ||||
|             case PackedResourceType::ShaderPack : | ||||
|                 // in theroy flame API can't do this but who knows, that *may* change ? | ||||
|                 // better to handle it if it *does* occure in the future | ||||
|   | ||||
| @@ -76,13 +76,8 @@ bool Flame::File::parseFromObject(const QJsonObject& obj,  bool throw_on_blocked | ||||
|     // It is also optional | ||||
|     type = File::Type::SingleFile; | ||||
|  | ||||
|     if (fileName.endsWith(".zip")) { | ||||
|         // this is probably a resource pack | ||||
|         targetFolder = "resourcepacks"; | ||||
|     } else { | ||||
|         // this is probably a mod, dunno what else could modpacks download | ||||
|     targetFolder = "mods"; | ||||
|     } | ||||
|  | ||||
|     // get the hash | ||||
|     hash = QString(); | ||||
|     auto hashes = Json::ensureArray(obj, "hashes"); | ||||
|   | ||||
							
								
								
									
										200
									
								
								launcher/modplatform/helpers/ExportToModList.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										200
									
								
								launcher/modplatform/helpers/ExportToModList.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,200 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, version 3. | ||||
|  * | ||||
|  *  This program is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| #include "ExportToModList.h" | ||||
| #include <QJsonArray> | ||||
| #include <QJsonDocument> | ||||
| #include <QJsonObject> | ||||
|  | ||||
| namespace ExportToModList { | ||||
| QString toHTML(QList<Mod*> mods, OptionalData extraData) | ||||
| { | ||||
|     QStringList lines; | ||||
|     for (auto mod : mods) { | ||||
|         auto meta = mod->metadata(); | ||||
|         auto modName = mod->name().toHtmlEscaped(); | ||||
|         if (extraData & Url) { | ||||
|             auto url = mod->metaurl().toHtmlEscaped(); | ||||
|             if (!url.isEmpty()) | ||||
|                 modName = QString("<a href=\"%1\">%2</a>").arg(url, modName); | ||||
|         } | ||||
|         auto line = modName; | ||||
|         if (extraData & Version) { | ||||
|             auto ver = mod->version(); | ||||
|             if (ver.isEmpty() && meta != nullptr) | ||||
|                 ver = meta->version().toString(); | ||||
|             if (!ver.isEmpty()) | ||||
|                 line += QString(" [%1]").arg(ver.toHtmlEscaped()); | ||||
|         } | ||||
|         if (extraData & Authors && !mod->authors().isEmpty()) | ||||
|             line += " by " + mod->authors().join(", ").toHtmlEscaped(); | ||||
|         lines.append(QString("<li>%1</li>").arg(line)); | ||||
|     } | ||||
|     return QString("<html><body><ul>\n\t%1\n</ul></body></html>").arg(lines.join("\n\t")); | ||||
| } | ||||
|  | ||||
| QString toMarkdown(QList<Mod*> mods, OptionalData extraData) | ||||
| { | ||||
|     QStringList lines; | ||||
|     for (auto mod : mods) { | ||||
|         auto meta = mod->metadata(); | ||||
|         auto modName = mod->name(); | ||||
|         if (extraData & Url) { | ||||
|             auto url = mod->metaurl(); | ||||
|             if (!url.isEmpty()) | ||||
|                 modName = QString("[%1](%2)").arg(modName, url); | ||||
|         } | ||||
|         auto line = modName; | ||||
|         if (extraData & Version) { | ||||
|             auto ver = mod->version(); | ||||
|             if (ver.isEmpty() && meta != nullptr) | ||||
|                 ver = meta->version().toString(); | ||||
|             if (!ver.isEmpty()) | ||||
|                 line += QString(" [%1]").arg(ver); | ||||
|         } | ||||
|         if (extraData & Authors && !mod->authors().isEmpty()) | ||||
|             line += " by " + mod->authors().join(", "); | ||||
|         lines << "- " + line; | ||||
|     } | ||||
|     return lines.join("\n"); | ||||
| } | ||||
|  | ||||
| QString toPlainTXT(QList<Mod*> mods, OptionalData extraData) | ||||
| { | ||||
|     QStringList lines; | ||||
|     for (auto mod : mods) { | ||||
|         auto meta = mod->metadata(); | ||||
|         auto modName = mod->name(); | ||||
|  | ||||
|         auto line = modName; | ||||
|         if (extraData & Url) { | ||||
|             auto url = mod->metaurl(); | ||||
|             if (!url.isEmpty()) | ||||
|                 line += QString(" (%1)").arg(url); | ||||
|         } | ||||
|         if (extraData & Version) { | ||||
|             auto ver = mod->version(); | ||||
|             if (ver.isEmpty() && meta != nullptr) | ||||
|                 ver = meta->version().toString(); | ||||
|             if (!ver.isEmpty()) | ||||
|                 line += QString(" [%1]").arg(ver); | ||||
|         } | ||||
|         if (extraData & Authors && !mod->authors().isEmpty()) | ||||
|             line += " by " + mod->authors().join(", "); | ||||
|         lines << line; | ||||
|     } | ||||
|     return lines.join("\n"); | ||||
| } | ||||
|  | ||||
| QString toJSON(QList<Mod*> mods, OptionalData extraData) | ||||
| { | ||||
|     QJsonArray lines; | ||||
|     for (auto mod : mods) { | ||||
|         auto meta = mod->metadata(); | ||||
|         auto modName = mod->name(); | ||||
|         QJsonObject line; | ||||
|         line["name"] = modName; | ||||
|         if (extraData & Url) { | ||||
|             auto url = mod->metaurl(); | ||||
|             if (!url.isEmpty()) | ||||
|                 line["url"] = url; | ||||
|         } | ||||
|         if (extraData & Version) { | ||||
|             auto ver = mod->version(); | ||||
|             if (ver.isEmpty() && meta != nullptr) | ||||
|                 ver = meta->version().toString(); | ||||
|             if (!ver.isEmpty()) | ||||
|                 line["version"] = ver; | ||||
|         } | ||||
|         if (extraData & Authors && !mod->authors().isEmpty()) | ||||
|             line["authors"] = QJsonArray::fromStringList(mod->authors()); | ||||
|         lines << line; | ||||
|     } | ||||
|     QJsonDocument doc; | ||||
|     doc.setArray(lines); | ||||
|     return doc.toJson(); | ||||
| } | ||||
|  | ||||
| QString toCSV(QList<Mod*> mods, OptionalData extraData) | ||||
| { | ||||
|     QStringList lines; | ||||
|     for (auto mod : mods) { | ||||
|         QStringList data; | ||||
|         auto meta = mod->metadata(); | ||||
|         auto modName = mod->name(); | ||||
|  | ||||
|         data << modName; | ||||
|         if (extraData & Url) | ||||
|             data << mod->metaurl(); | ||||
|         if (extraData & Version) { | ||||
|             auto ver = mod->version(); | ||||
|             if (ver.isEmpty() && meta != nullptr) | ||||
|                 ver = meta->version().toString(); | ||||
|             data << ver; | ||||
|         } | ||||
|         if (extraData & Authors) { | ||||
|             QString authors; | ||||
|             if (mod->authors().length() == 1) | ||||
|                 authors = mod->authors().back(); | ||||
|             else if (mod->authors().length() > 1) | ||||
|                 authors = QString("\"%1\"").arg(mod->authors().join(",")); | ||||
|             data << authors; | ||||
|         } | ||||
|         lines << data.join(","); | ||||
|     } | ||||
|     return lines.join("\n"); | ||||
| } | ||||
|  | ||||
| QString exportToModList(QList<Mod*> mods, Formats format, OptionalData extraData) | ||||
| { | ||||
|     switch (format) { | ||||
|         case HTML: | ||||
|             return toHTML(mods, extraData); | ||||
|         case MARKDOWN: | ||||
|             return toMarkdown(mods, extraData); | ||||
|         case PLAINTXT: | ||||
|             return toPlainTXT(mods, extraData); | ||||
|         case JSON: | ||||
|             return toJSON(mods, extraData); | ||||
|         case CSV: | ||||
|             return toCSV(mods, extraData); | ||||
|         default: { | ||||
|             return QString("unknown format:%1").arg(format); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| QString exportToModList(QList<Mod*> mods, QString lineTemplate) | ||||
| { | ||||
|     QStringList lines; | ||||
|     for (auto mod : mods) { | ||||
|         auto meta = mod->metadata(); | ||||
|         auto modName = mod->name(); | ||||
|         auto url = mod->metaurl(); | ||||
|         auto ver = mod->version(); | ||||
|         if (ver.isEmpty() && meta != nullptr) | ||||
|             ver = meta->version().toString(); | ||||
|         auto authors = mod->authors().join(", "); | ||||
|         lines << QString(lineTemplate) | ||||
|                      .replace("{name}", modName) | ||||
|                      .replace("{url}", url) | ||||
|                      .replace("{version}", ver) | ||||
|                      .replace("{authors}", authors); | ||||
|     } | ||||
|     return lines.join("\n"); | ||||
| } | ||||
| }  // namespace ExportToModList | ||||
							
								
								
									
										33
									
								
								launcher/modplatform/helpers/ExportToModList.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								launcher/modplatform/helpers/ExportToModList.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, version 3. | ||||
|  * | ||||
|  *  This program is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| #pragma once | ||||
| #include <QList> | ||||
| #include <QString> | ||||
| #include "minecraft/mod/Mod.h" | ||||
|  | ||||
| namespace ExportToModList { | ||||
|  | ||||
| enum Formats { HTML, MARKDOWN, PLAINTXT, JSON, CSV, CUSTOM }; | ||||
| enum OptionalData { | ||||
|     Authors = 1 << 0, | ||||
|     Url = 1 << 1, | ||||
|     Version = 1 << 2, | ||||
| }; | ||||
| QString exportToModList(QList<Mod*> mods, Formats format, OptionalData extraData); | ||||
| QString exportToModList(QList<Mod*> mods, QString lineTemplate); | ||||
| }  // namespace ExportToModList | ||||
| @@ -37,16 +37,16 @@ | ||||
|  | ||||
| #include <QtConcurrent> | ||||
|  | ||||
| #include "MMCZip.h" | ||||
| #include "BaseInstance.h" | ||||
| #include "FileSystem.h" | ||||
| #include "settings/INISettingsObject.h" | ||||
| #include "MMCZip.h" | ||||
| #include "minecraft/GradleSpecifier.h" | ||||
| #include "minecraft/MinecraftInstance.h" | ||||
| #include "minecraft/PackProfile.h" | ||||
| #include "minecraft/GradleSpecifier.h" | ||||
| #include "settings/INISettingsObject.h" | ||||
|  | ||||
| #include "BuildConfig.h" | ||||
| #include "Application.h" | ||||
| #include "BuildConfig.h" | ||||
|  | ||||
| namespace LegacyFTB { | ||||
|  | ||||
| @@ -65,6 +65,7 @@ void PackInstallTask::executeTask() | ||||
| void PackInstallTask::downloadPack() | ||||
| { | ||||
|     setStatus(tr("Downloading zip for %1").arg(m_pack.name)); | ||||
|     setProgress(1, 4); | ||||
|     setAbortable(false); | ||||
|  | ||||
|     archivePath = QString("%1/%2/%3").arg(m_pack.dir, m_version.replace(".", "_"), m_pack.file); | ||||
| @@ -78,11 +79,10 @@ void PackInstallTask::downloadPack() | ||||
|     } | ||||
|     netJobContainer->addNetAction(Net::Download::makeFile(url, archivePath)); | ||||
|  | ||||
|     connect(netJobContainer.get(), &NetJob::succeeded, this, &PackInstallTask::onDownloadSucceeded); | ||||
|     connect(netJobContainer.get(), &NetJob::failed, this, &PackInstallTask::onDownloadFailed); | ||||
|     connect(netJobContainer.get(), &NetJob::progress, this, &PackInstallTask::onDownloadProgress); | ||||
|     connect(netJobContainer.get(), &NetJob::succeeded, this, &PackInstallTask::unzip); | ||||
|     connect(netJobContainer.get(), &NetJob::failed, this, &PackInstallTask::emitFailed); | ||||
|     connect(netJobContainer.get(), &NetJob::stepProgress, this, &PackInstallTask::propogateStepProgress); | ||||
|     connect(netJobContainer.get(), &NetJob::aborted, this, &PackInstallTask::onDownloadAborted); | ||||
|     connect(netJobContainer.get(), &NetJob::aborted, this, &PackInstallTask::emitAborted); | ||||
|  | ||||
|     netJobContainer->start(); | ||||
|  | ||||
| @@ -90,27 +90,6 @@ void PackInstallTask::downloadPack() | ||||
|     progress(1, 4); | ||||
| } | ||||
|  | ||||
| void PackInstallTask::onDownloadSucceeded() | ||||
| { | ||||
|     unzip(); | ||||
| } | ||||
|  | ||||
| void PackInstallTask::onDownloadFailed(QString reason) | ||||
| { | ||||
|     emitFailed(reason); | ||||
| } | ||||
|  | ||||
| void PackInstallTask::onDownloadProgress(qint64 current, qint64 total) | ||||
| { | ||||
|     progress(current, total * 4); | ||||
|     setStatus(tr("Downloading zip for %1 (%2%)").arg(m_pack.name).arg(current / 10)); | ||||
| } | ||||
|  | ||||
| void PackInstallTask::onDownloadAborted() | ||||
| { | ||||
|     emitAborted(); | ||||
| } | ||||
|  | ||||
| void PackInstallTask::unzip() | ||||
| { | ||||
|     setStatus(tr("Extracting modpack")); | ||||
| @@ -120,16 +99,17 @@ void PackInstallTask::unzip() | ||||
|     QDir extractDir(m_stagingPath); | ||||
|  | ||||
|     m_packZip.reset(new QuaZip(archivePath)); | ||||
|     if(!m_packZip->open(QuaZip::mdUnzip)) | ||||
|     { | ||||
|     if (!m_packZip->open(QuaZip::mdUnzip)) { | ||||
|         emitFailed(tr("Failed to open modpack file %1!").arg(archivePath)); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
| #if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0) | ||||
|     m_extractFuture = QtConcurrent::run(QThreadPool::globalInstance(), QOverload<QString, QString>::of(MMCZip::extractDir), archivePath, extractDir.absolutePath() + "/unzip"); | ||||
|     m_extractFuture = QtConcurrent::run(QThreadPool::globalInstance(), QOverload<QString, QString>::of(MMCZip::extractDir), archivePath, | ||||
|                                         extractDir.absolutePath() + "/unzip"); | ||||
| #else | ||||
|     m_extractFuture = QtConcurrent::run(QThreadPool::globalInstance(), MMCZip::extractDir, archivePath, extractDir.absolutePath() + "/unzip"); | ||||
|     m_extractFuture = | ||||
|         QtConcurrent::run(QThreadPool::globalInstance(), MMCZip::extractDir, archivePath, extractDir.absolutePath() + "/unzip"); | ||||
| #endif | ||||
|     connect(&m_extractFutureWatcher, &QFutureWatcher<QStringList>::finished, this, &PackInstallTask::onUnzipFinished); | ||||
|     connect(&m_extractFutureWatcher, &QFutureWatcher<QStringList>::canceled, this, &PackInstallTask::onUnzipCanceled); | ||||
| @@ -151,11 +131,9 @@ void PackInstallTask::install() | ||||
|     setStatus(tr("Installing modpack")); | ||||
|     progress(3, 4); | ||||
|     QDir unzipMcDir(m_stagingPath + "/unzip/minecraft"); | ||||
|     if(unzipMcDir.exists()) | ||||
|     { | ||||
|         //ok, found minecraft dir, move contents to instance dir | ||||
|         if(!QDir().rename(m_stagingPath + "/unzip/minecraft", m_stagingPath + "/.minecraft")) | ||||
|         { | ||||
|     if (unzipMcDir.exists()) { | ||||
|         // ok, found minecraft dir, move contents to instance dir | ||||
|         if (!QDir().rename(m_stagingPath + "/unzip/minecraft", m_stagingPath + "/.minecraft")) { | ||||
|             emitFailed(tr("Failed to move unzipped Minecraft!")); | ||||
|             return; | ||||
|         } | ||||
| @@ -172,23 +150,20 @@ void PackInstallTask::install() | ||||
|  | ||||
|     bool fallback = true; | ||||
|  | ||||
|     //handle different versions | ||||
|     // handle different versions | ||||
|     QFile packJson(m_stagingPath + "/.minecraft/pack.json"); | ||||
|     QDir jarmodDir = QDir(m_stagingPath + "/unzip/instMods"); | ||||
|     if(packJson.exists()) | ||||
|     { | ||||
|     if (packJson.exists()) { | ||||
|         packJson.open(QIODevice::ReadOnly | QIODevice::Text); | ||||
|         QJsonDocument doc = QJsonDocument::fromJson(packJson.readAll()); | ||||
|         packJson.close(); | ||||
|  | ||||
|         //we only care about the libs | ||||
|         // we only care about the libs | ||||
|         QJsonArray libs = doc.object().value("libraries").toArray(); | ||||
|  | ||||
|         foreach (const QJsonValue &value, libs) | ||||
|         { | ||||
|         foreach (const QJsonValue& value, libs) { | ||||
|             QString nameValue = value.toObject().value("name").toString(); | ||||
|             if(!nameValue.startsWith("net.minecraftforge")) | ||||
|             { | ||||
|             if (!nameValue.startsWith("net.minecraftforge")) { | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
| @@ -199,16 +174,13 @@ void PackInstallTask::install() | ||||
|             fallback = false; | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|     } | ||||
|  | ||||
|     if(jarmodDir.exists()) | ||||
|     { | ||||
|     if (jarmodDir.exists()) { | ||||
|         qDebug() << "Found jarmods, installing..."; | ||||
|  | ||||
|         QStringList jarmods; | ||||
|         for (auto info: jarmodDir.entryInfoList(QDir::NoDotAndDotDot | QDir::Files)) | ||||
|         { | ||||
|         for (auto info : jarmodDir.entryInfoList(QDir::NoDotAndDotDot | QDir::Files)) { | ||||
|             qDebug() << "Jarmod:" << info.fileName(); | ||||
|             jarmods.push_back(info.absoluteFilePath()); | ||||
|         } | ||||
| @@ -217,12 +189,11 @@ void PackInstallTask::install() | ||||
|         fallback = false; | ||||
|     } | ||||
|  | ||||
|     //just nuke unzip directory, it s not needed anymore | ||||
|     // just nuke unzip directory, it s not needed anymore | ||||
|     FS::deletePath(m_stagingPath + "/unzip"); | ||||
|  | ||||
|     if(fallback) | ||||
|     { | ||||
|         //TODO: Some fallback mechanism... or just keep failing! | ||||
|     if (fallback) { | ||||
|         // TODO: Some fallback mechanism... or just keep failing! | ||||
|         emitFailed(tr("No installation method found!")); | ||||
|         return; | ||||
|     } | ||||
| @@ -232,8 +203,7 @@ void PackInstallTask::install() | ||||
|     progress(4, 4); | ||||
|  | ||||
|     instance.setName(name()); | ||||
|     if(m_instIcon == "default") | ||||
|     { | ||||
|     if (m_instIcon == "default") { | ||||
|         m_instIcon = "ftb_logo"; | ||||
|     } | ||||
|     instance.setIconKey(m_instIcon); | ||||
| @@ -252,4 +222,4 @@ bool PackInstallTask::abort() | ||||
|     return InstanceTask::abort(); | ||||
| } | ||||
|  | ||||
| } | ||||
| }  // namespace LegacyFTB | ||||
|   | ||||
| @@ -1,12 +1,12 @@ | ||||
| #pragma once | ||||
| #include "InstanceTask.h" | ||||
| #include "net/NetJob.h" | ||||
| #include <quazip/quazip.h> | ||||
| #include <quazip/quazipdir.h> | ||||
| #include "InstanceTask.h" | ||||
| #include "PackHelpers.h" | ||||
| #include "meta/Index.h" | ||||
| #include "meta/Version.h" | ||||
| #include "meta/VersionList.h" | ||||
| #include "PackHelpers.h" | ||||
| #include "net/NetJob.h" | ||||
|  | ||||
| #include "net/NetJob.h" | ||||
|  | ||||
| @@ -14,36 +14,31 @@ | ||||
|  | ||||
| namespace LegacyFTB { | ||||
|  | ||||
| class PackInstallTask : public InstanceTask | ||||
| { | ||||
| class PackInstallTask : public InstanceTask { | ||||
|     Q_OBJECT | ||||
|  | ||||
| public: | ||||
|    public: | ||||
|     explicit PackInstallTask(shared_qobject_ptr<QNetworkAccessManager> network, Modpack pack, QString version); | ||||
|     virtual ~PackInstallTask(){} | ||||
|     virtual ~PackInstallTask() {} | ||||
|  | ||||
|     bool canAbort() const override { return true; } | ||||
|     bool abort() override; | ||||
|  | ||||
| protected: | ||||
|    protected: | ||||
|     //! Entry point for tasks. | ||||
|     virtual void executeTask() override; | ||||
|  | ||||
| private: | ||||
|    private: | ||||
|     void downloadPack(); | ||||
|     void unzip(); | ||||
|     void install(); | ||||
|  | ||||
| private slots: | ||||
|     void onDownloadSucceeded(); | ||||
|     void onDownloadFailed(QString reason); | ||||
|     void onDownloadProgress(qint64 current, qint64 total); | ||||
|     void onDownloadAborted(); | ||||
|    private slots: | ||||
|  | ||||
|     void onUnzipFinished(); | ||||
|     void onUnzipCanceled(); | ||||
|  | ||||
| private: /* data */ | ||||
|    private: /* data */ | ||||
|     shared_qobject_ptr<QNetworkAccessManager> m_network; | ||||
|     bool abortable = false; | ||||
|     std::unique_ptr<QuaZip> m_packZip; | ||||
| @@ -56,4 +51,4 @@ private: /* data */ | ||||
|     QString m_version; | ||||
| }; | ||||
|  | ||||
| } | ||||
| }  // namespace LegacyFTB | ||||
|   | ||||
| @@ -42,6 +42,7 @@ | ||||
| #include <QDir> | ||||
| #include <QLibraryInfo> | ||||
| #include <QDebug> | ||||
| #include <locale> | ||||
|  | ||||
| #include "FileSystem.h" | ||||
| #include "net/NetJob.h" | ||||
| @@ -527,34 +528,34 @@ Language * TranslationsModel::findLanguage(const QString& key) | ||||
|     } | ||||
| } | ||||
|  | ||||
| void TranslationsModel::setUseSystemLocale(bool useSystemLocale) | ||||
| { | ||||
|     APPLICATION->settings()->set("UseSystemLocale", useSystemLocale); | ||||
|     QLocale::setDefault(QLocale(useSystemLocale ? QString::fromStdString(std::locale().name()) : defaultLangCode)); | ||||
| } | ||||
|  | ||||
| bool TranslationsModel::selectLanguage(QString key) | ||||
| { | ||||
|     QString &langCode = key; | ||||
|     QString& langCode = key; | ||||
|     auto langPtr = findLanguage(key); | ||||
|  | ||||
|     if (langCode.isEmpty()) | ||||
|     { | ||||
|     if (langCode.isEmpty()) { | ||||
|         d->no_language_set = true; | ||||
|     } | ||||
|  | ||||
|     if(!langPtr) | ||||
|     { | ||||
|     if (!langPtr) { | ||||
|         qWarning() << "Selected invalid language" << key << ", defaulting to" << defaultLangCode; | ||||
|         langCode = defaultLangCode; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|     } else { | ||||
|         langCode = langPtr->key; | ||||
|     } | ||||
|  | ||||
|     // uninstall existing translators if there are any | ||||
|     if (d->m_app_translator) | ||||
|     { | ||||
|     if (d->m_app_translator) { | ||||
|         QCoreApplication::removeTranslator(d->m_app_translator.get()); | ||||
|         d->m_app_translator.reset(); | ||||
|     } | ||||
|     if (d->m_qt_translator) | ||||
|     { | ||||
|     if (d->m_qt_translator) { | ||||
|         QCoreApplication::removeTranslator(d->m_qt_translator.get()); | ||||
|         d->m_qt_translator.reset(); | ||||
|     } | ||||
| @@ -564,8 +565,9 @@ bool TranslationsModel::selectLanguage(QString key) | ||||
|      * In a multithreaded application, the default locale should be set at application startup, before any non-GUI threads are created. | ||||
|      * This function is not reentrant. | ||||
|      */ | ||||
|     QLocale locale = QLocale(langCode); | ||||
|     QLocale::setDefault(locale); | ||||
|     QLocale::setDefault( | ||||
|         QLocale(APPLICATION->settings()->get("UseSystemLocale").toBool() ? QString::fromStdString(std::locale().name()) : langCode)); | ||||
|  | ||||
|  | ||||
|     // if it's the default UI language, finish | ||||
|     if(langCode == defaultLangCode) | ||||
|   | ||||
| @@ -20,17 +20,16 @@ | ||||
|  | ||||
| struct Language; | ||||
|  | ||||
| class TranslationsModel : public QAbstractListModel | ||||
| { | ||||
| class TranslationsModel : public QAbstractListModel { | ||||
|     Q_OBJECT | ||||
| public: | ||||
|     explicit TranslationsModel(QString path, QObject *parent = 0); | ||||
|    public: | ||||
|     explicit TranslationsModel(QString path, QObject* parent = 0); | ||||
|     virtual ~TranslationsModel(); | ||||
|  | ||||
|     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) const override; | ||||
|     int rowCount(const QModelIndex &parent = QModelIndex()) const override; | ||||
|     int columnCount(const QModelIndex & parent) const override; | ||||
|     int rowCount(const QModelIndex& parent = QModelIndex()) const override; | ||||
|     int columnCount(const QModelIndex& parent) const override; | ||||
|  | ||||
|     bool selectLanguage(QString key); | ||||
|     void updateLanguage(QString key); | ||||
| @@ -38,27 +37,27 @@ public: | ||||
|     QString selectedLanguage(); | ||||
|  | ||||
|     void downloadIndex(); | ||||
|     void setUseSystemLocale(bool useSystemLocale); | ||||
|  | ||||
| private: | ||||
|     Language *findLanguage(const QString & key); | ||||
|    private: | ||||
|     Language* findLanguage(const QString& key); | ||||
|     void reloadLocalFiles(); | ||||
|     void downloadTranslation(QString key); | ||||
|     void downloadNext(); | ||||
|  | ||||
|     // hide copy constructor | ||||
|     TranslationsModel(const TranslationsModel &) = delete; | ||||
|     TranslationsModel(const TranslationsModel&) = delete; | ||||
|     // hide assign op | ||||
|     TranslationsModel &operator=(const TranslationsModel &) = delete; | ||||
|     TranslationsModel& operator=(const TranslationsModel&) = delete; | ||||
|  | ||||
| private slots: | ||||
|    private slots: | ||||
|     void indexReceived(); | ||||
|     void indexFailed(QString reason); | ||||
|     void dlFailed(QString reason); | ||||
|     void dlGood(); | ||||
|     void translationDirChanged(const QString &path); | ||||
|     void translationDirChanged(const QString& path); | ||||
|  | ||||
|  | ||||
| private: /* data */ | ||||
|    private: /* data */ | ||||
|     struct Private; | ||||
|     std::unique_ptr<Private> d; | ||||
| }; | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -158,6 +158,7 @@ private slots: | ||||
|     void on_actionExportInstanceZip_triggered(); | ||||
|     void on_actionExportInstanceMrPack_triggered(); | ||||
|     void on_actionExportInstanceFlamePack_triggered(); | ||||
|     void on_actionExportInstanceToModList_triggered(); | ||||
|  | ||||
|     void on_actionRenameInstance_triggered(); | ||||
|  | ||||
|   | ||||
| @@ -487,6 +487,14 @@ | ||||
|     <string>CurseForge (zip)</string>   | ||||
|    </property> | ||||
|   </action> | ||||
|   <action name="actionExportInstanceToModList"> | ||||
|    <property name="icon"> | ||||
|     <iconset theme="new"/> | ||||
|    </property> | ||||
|    <property name="text"> | ||||
|     <string>Mod List</string> | ||||
|    </property> | ||||
|   </action> | ||||
|   <action name="actionCreateInstanceShortcut"> | ||||
|    <property name="icon"> | ||||
|     <iconset theme="shortcut"> | ||||
|   | ||||
| @@ -3,6 +3,7 @@ | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net> | ||||
|  *  Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me> | ||||
|  *  Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
| @@ -41,6 +42,9 @@ | ||||
| #include <QFileSystemModel> | ||||
| #include <QMessageBox> | ||||
| #include "FileIgnoreProxy.h" | ||||
| #include "QObjectPtr.h" | ||||
| #include "ui/dialogs/CustomMessageBox.h" | ||||
| #include "ui/dialogs/ProgressDialog.h" | ||||
| #include "ui_ExportInstanceDialog.h" | ||||
|  | ||||
| #include <FileSystem.h> | ||||
| @@ -72,7 +76,7 @@ ExportInstanceDialog::ExportInstanceDialog(InstancePtr instance, QWidget* parent | ||||
|     ui->treeView->setRootIndex(proxyModel->mapFromSource(model->index(root))); | ||||
|     ui->treeView->sortByColumn(0, Qt::AscendingOrder); | ||||
|  | ||||
|     connect(proxyModel, SIGNAL(rowsInserted(QModelIndex,int,int)), SLOT(rowsInserted(QModelIndex,int,int))); | ||||
|     connect(proxyModel, SIGNAL(rowsInserted(QModelIndex, int, int)), SLOT(rowsInserted(QModelIndex, int, int))); | ||||
|  | ||||
|     model->setFilter(QDir::AllEntries | QDir::NoDotAndDotDot | QDir::AllDirs | QDir::Hidden); | ||||
|     model->setRootPath(root); | ||||
| @@ -92,32 +96,26 @@ void SaveIcon(InstancePtr m_instance) | ||||
|     auto iconKey = m_instance->iconKey(); | ||||
|     auto iconList = APPLICATION->icons(); | ||||
|     auto mmcIcon = iconList->icon(iconKey); | ||||
|     if(!mmcIcon || mmcIcon->isBuiltIn()) { | ||||
|     if (!mmcIcon || mmcIcon->isBuiltIn()) { | ||||
|         return; | ||||
|     } | ||||
|     auto path = mmcIcon->getFilePath(); | ||||
|     if(!path.isNull()) { | ||||
|         QFileInfo inInfo (path); | ||||
|         FS::copy(path, FS::PathCombine(m_instance->instanceRoot(), inInfo.fileName())) (); | ||||
|     if (!path.isNull()) { | ||||
|         QFileInfo inInfo(path); | ||||
|         FS::copy(path, FS::PathCombine(m_instance->instanceRoot(), inInfo.fileName()))(); | ||||
|         return; | ||||
|     } | ||||
|     auto & image = mmcIcon->m_images[mmcIcon->type()]; | ||||
|     auto & icon = image.icon; | ||||
|     auto& image = mmcIcon->m_images[mmcIcon->type()]; | ||||
|     auto& icon = image.icon; | ||||
|     auto sizes = icon.availableSizes(); | ||||
|     if(sizes.size() == 0) | ||||
|     { | ||||
|     if (sizes.size() == 0) { | ||||
|         return; | ||||
|     } | ||||
|     auto areaOf = [](QSize size) | ||||
|     { | ||||
|         return size.width() * size.height(); | ||||
|     }; | ||||
|     auto areaOf = [](QSize size) { return size.width() * size.height(); }; | ||||
|     QSize largest = sizes[0]; | ||||
|     // find variant with largest area | ||||
|     for(auto size: sizes) | ||||
|     { | ||||
|         if(areaOf(largest) < areaOf(size)) | ||||
|         { | ||||
|     for (auto size : sizes) { | ||||
|         if (areaOf(largest) < areaOf(size)) { | ||||
|             largest = size; | ||||
|         } | ||||
|     } | ||||
| @@ -125,16 +123,15 @@ void SaveIcon(InstancePtr m_instance) | ||||
|     pixmap.save(FS::PathCombine(m_instance->instanceRoot(), iconKey + ".png")); | ||||
| } | ||||
|  | ||||
| bool ExportInstanceDialog::doExport() | ||||
| void ExportInstanceDialog::doExport() | ||||
| { | ||||
|     auto name = FS::RemoveInvalidFilenameChars(m_instance->name()); | ||||
|  | ||||
|     const QString output = QFileDialog::getSaveFileName( | ||||
|         this, tr("Export %1").arg(m_instance->name()), | ||||
|     const QString output = QFileDialog::getSaveFileName(this, tr("Export %1").arg(m_instance->name()), | ||||
|                                                         FS::PathCombine(QDir::homePath(), name + ".zip"), "Zip (*.zip)", nullptr); | ||||
|     if (output.isEmpty()) | ||||
|     { | ||||
|         return false; | ||||
|     if (output.isEmpty()) { | ||||
|         QDialog::done(QDialog::Rejected); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     SaveIcon(m_instance); | ||||
| @@ -143,46 +140,40 @@ bool ExportInstanceDialog::doExport() | ||||
|     if (!MMCZip::collectFileListRecursively(m_instance->instanceRoot(), nullptr, &files, | ||||
|                                             std::bind(&FileIgnoreProxy::filterFile, proxyModel, std::placeholders::_1))) { | ||||
|         QMessageBox::warning(this, tr("Error"), tr("Unable to export instance")); | ||||
|         return false; | ||||
|         QDialog::done(QDialog::Rejected); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!MMCZip::compressDirFiles(output, m_instance->instanceRoot(), files, true)) | ||||
|     { | ||||
|         QMessageBox::warning(this, tr("Error"), tr("Unable to export instance")); | ||||
|         return false; | ||||
|     } | ||||
|     return true; | ||||
|     auto task = makeShared<MMCZip::ExportToZipTask>(output, m_instance->instanceRoot(), files, "", true); | ||||
|  | ||||
|     connect(task.get(), &Task::failed, this, | ||||
|             [this, output](QString reason) { CustomMessageBox::selectable(this, tr("Error"), reason, QMessageBox::Critical)->show(); }); | ||||
|     connect(task.get(), &Task::finished, this, [task] { task->deleteLater(); }); | ||||
|  | ||||
|     ProgressDialog progress(this); | ||||
|     progress.setSkipButton(true, tr("Abort")); | ||||
|     auto result = progress.execWithTask(task.get()); | ||||
|     QDialog::done(result); | ||||
| } | ||||
|  | ||||
| void ExportInstanceDialog::done(int result) | ||||
| { | ||||
|     savePackIgnore(); | ||||
|     if (result == QDialog::Accepted) | ||||
|     { | ||||
|         if (doExport()) | ||||
|         { | ||||
|             QDialog::done(QDialog::Accepted); | ||||
|     if (result == QDialog::Accepted) { | ||||
|         doExport(); | ||||
|         return; | ||||
|     } | ||||
|         else | ||||
|         { | ||||
|             return; | ||||
|         } | ||||
|     } | ||||
|     QDialog::done(result); | ||||
| } | ||||
|  | ||||
| void ExportInstanceDialog::rowsInserted(QModelIndex parent, int top, int bottom) | ||||
| { | ||||
|     //WARNING: possible off-by-one? | ||||
|     for(int i = top; i < bottom; i++) | ||||
|     { | ||||
|     // WARNING: possible off-by-one? | ||||
|     for (int i = top; i < bottom; i++) { | ||||
|         auto node = proxyModel->index(i, 0, parent); | ||||
|         if(proxyModel->shouldExpand(node)) | ||||
|         { | ||||
|         if (proxyModel->shouldExpand(node)) { | ||||
|             auto expNode = node.parent(); | ||||
|             if(!expNode.isValid()) | ||||
|             { | ||||
|             if (!expNode.isValid()) { | ||||
|                 continue; | ||||
|             } | ||||
|             ui->treeView->expand(node); | ||||
| @@ -199,8 +190,7 @@ void ExportInstanceDialog::loadPackIgnore() | ||||
| { | ||||
|     auto filename = ignoreFileName(); | ||||
|     QFile ignoreFile(filename); | ||||
|     if(!ignoreFile.open(QIODevice::ReadOnly)) | ||||
|     { | ||||
|     if (!ignoreFile.open(QIODevice::ReadOnly)) { | ||||
|         return; | ||||
|     } | ||||
|     auto data = ignoreFile.readAll(); | ||||
| @@ -216,12 +206,9 @@ void ExportInstanceDialog::savePackIgnore() | ||||
| { | ||||
|     auto data = proxyModel->blockedPaths().toStringList().join('\n').toUtf8(); | ||||
|     auto filename = ignoreFileName(); | ||||
|     try | ||||
|     { | ||||
|     try { | ||||
|         FS::write(filename, data); | ||||
|     } | ||||
|     catch (const Exception &e) | ||||
|     { | ||||
|     } catch (const Exception& e) { | ||||
|         qWarning() << e.cause(); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me> | ||||
|  *  Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
| @@ -38,39 +39,37 @@ | ||||
| #include <QDialog> | ||||
| #include <QModelIndex> | ||||
| #include <memory> | ||||
| #include "FileIgnoreProxy.h" | ||||
| #include "FastFileIconProvider.h" | ||||
| #include "FileIgnoreProxy.h" | ||||
|  | ||||
| class BaseInstance; | ||||
| typedef std::shared_ptr<BaseInstance> InstancePtr; | ||||
|  | ||||
| namespace Ui | ||||
| { | ||||
| namespace Ui { | ||||
| class ExportInstanceDialog; | ||||
| } | ||||
|  | ||||
| class ExportInstanceDialog : public QDialog | ||||
| { | ||||
| class ExportInstanceDialog : public QDialog { | ||||
|     Q_OBJECT | ||||
|  | ||||
| public: | ||||
|     explicit ExportInstanceDialog(InstancePtr instance, QWidget *parent = 0); | ||||
|    public: | ||||
|     explicit ExportInstanceDialog(InstancePtr instance, QWidget* parent = 0); | ||||
|     ~ExportInstanceDialog(); | ||||
|  | ||||
|     virtual void done(int result); | ||||
|  | ||||
| private: | ||||
|     bool doExport(); | ||||
|    private: | ||||
|     void doExport(); | ||||
|     void loadPackIgnore(); | ||||
|     void savePackIgnore(); | ||||
|     QString ignoreFileName(); | ||||
|  | ||||
| private: | ||||
|     Ui::ExportInstanceDialog *ui; | ||||
|    private: | ||||
|     Ui::ExportInstanceDialog* ui; | ||||
|     InstancePtr m_instance; | ||||
|     FileIgnoreProxy * proxyModel; | ||||
|     FileIgnoreProxy* proxyModel; | ||||
|     FastFileIconProvider icons; | ||||
|  | ||||
| private slots: | ||||
|    private slots: | ||||
|     void rowsInserted(QModelIndex parent, int top, int bottom); | ||||
| }; | ||||
|   | ||||
							
								
								
									
										223
									
								
								launcher/ui/dialogs/ExportToModListDialog.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										223
									
								
								launcher/ui/dialogs/ExportToModListDialog.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,223 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, version 3. | ||||
|  * | ||||
|  *  This program is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
|  | ||||
| #include "ExportToModListDialog.h" | ||||
| #include <QCheckBox> | ||||
| #include <QComboBox> | ||||
| #include <QTextEdit> | ||||
| #include "FileSystem.h" | ||||
| #include "Markdown.h" | ||||
| #include "minecraft/MinecraftInstance.h" | ||||
| #include "minecraft/mod/ModFolderModel.h" | ||||
| #include "modplatform/helpers/ExportToModList.h" | ||||
| #include "ui_ExportToModListDialog.h" | ||||
|  | ||||
| #include <QFileDialog> | ||||
| #include <QFileSystemModel> | ||||
| #include <QJsonDocument> | ||||
| #include <QMessageBox> | ||||
| #include <QPushButton> | ||||
|  | ||||
| const QHash<ExportToModList::Formats, QString> ExportToModListDialog::exampleLines = { | ||||
|     { ExportToModList::HTML, "<li><a href=\"{url}\">{name}</a> [{version}] by {authors}</li>" }, | ||||
|     { ExportToModList::MARKDOWN, "[{name}]({url}) [{version}] by {authors}" }, | ||||
|     { ExportToModList::PLAINTXT, "{name} ({url}) [{version}] by {authors}" }, | ||||
|     { ExportToModList::JSON, "{\"name\":\"{name}\",\"url\":\"{url}\",\"version\":\"{version}\",\"authors\":\"{authors}\"}," }, | ||||
|     { ExportToModList::CSV, "{name},{url},{version},\"{authors}\"" }, | ||||
| }; | ||||
|  | ||||
| ExportToModListDialog::ExportToModListDialog(InstancePtr instance, QWidget* parent) | ||||
|     : QDialog(parent), m_template_changed(false), name(instance->name()), ui(new Ui::ExportToModListDialog) | ||||
| { | ||||
|     ui->setupUi(this); | ||||
|     enableCustom(false); | ||||
|  | ||||
|     MinecraftInstance* mcInstance = dynamic_cast<MinecraftInstance*>(instance.get()); | ||||
|     if (mcInstance) { | ||||
|         mcInstance->loaderModList()->update(); | ||||
|         connect(mcInstance->loaderModList().get(), &ModFolderModel::updateFinished, this, [this, mcInstance]() { | ||||
|             m_allMods = mcInstance->loaderModList()->allMods(); | ||||
|             triggerImp(); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     connect(ui->formatComboBox, QOverload<int>::of(&QComboBox::currentIndexChanged), this, &ExportToModListDialog::formatChanged); | ||||
|     connect(ui->authorsCheckBox, &QCheckBox::stateChanged, this, &ExportToModListDialog::trigger); | ||||
|     connect(ui->versionCheckBox, &QCheckBox::stateChanged, this, &ExportToModListDialog::trigger); | ||||
|     connect(ui->urlCheckBox, &QCheckBox::stateChanged, this, &ExportToModListDialog::trigger); | ||||
|     connect(ui->authorsButton, &QPushButton::clicked, this, [this](bool) { addExtra(ExportToModList::Authors); }); | ||||
|     connect(ui->versionButton, &QPushButton::clicked, this, [this](bool) { addExtra(ExportToModList::Version); }); | ||||
|     connect(ui->urlButton, &QPushButton::clicked, this, [this](bool) { addExtra(ExportToModList::Url); }); | ||||
|     connect(ui->templateText, &QTextEdit::textChanged, this, [this] { | ||||
|         if (ui->templateText->toPlainText() != exampleLines[format]) | ||||
|             ui->formatComboBox->setCurrentIndex(5); | ||||
|         else | ||||
|             triggerImp(); | ||||
|     }); | ||||
|     connect(ui->copyButton, &QPushButton::clicked, this, [this](bool) { | ||||
|         this->ui->finalText->selectAll(); | ||||
|         this->ui->finalText->copy(); | ||||
|     }); | ||||
| } | ||||
|  | ||||
| ExportToModListDialog::~ExportToModListDialog() | ||||
| { | ||||
|     delete ui; | ||||
| } | ||||
|  | ||||
| void ExportToModListDialog::formatChanged(int index) | ||||
| { | ||||
|     switch (index) { | ||||
|         case 0: { | ||||
|             enableCustom(false); | ||||
|             ui->resultText->show(); | ||||
|             format = ExportToModList::HTML; | ||||
|             break; | ||||
|         } | ||||
|         case 1: { | ||||
|             enableCustom(false); | ||||
|             ui->resultText->show(); | ||||
|             format = ExportToModList::MARKDOWN; | ||||
|             break; | ||||
|         } | ||||
|         case 2: { | ||||
|             enableCustom(false); | ||||
|             ui->resultText->hide(); | ||||
|             format = ExportToModList::PLAINTXT; | ||||
|             break; | ||||
|         } | ||||
|         case 3: { | ||||
|             enableCustom(false); | ||||
|             ui->resultText->hide(); | ||||
|             format = ExportToModList::JSON; | ||||
|             break; | ||||
|         } | ||||
|         case 4: { | ||||
|             enableCustom(false); | ||||
|             ui->resultText->hide(); | ||||
|             format = ExportToModList::CSV; | ||||
|             break; | ||||
|         } | ||||
|         case 5: { | ||||
|             m_template_changed = true; | ||||
|             enableCustom(true); | ||||
|             ui->resultText->hide(); | ||||
|             format = ExportToModList::CUSTOM; | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|     triggerImp(); | ||||
| } | ||||
|  | ||||
| void ExportToModListDialog::triggerImp() | ||||
| { | ||||
|     if (format == ExportToModList::CUSTOM) { | ||||
|         ui->finalText->setPlainText(ExportToModList::exportToModList(m_allMods, ui->templateText->toPlainText())); | ||||
|         return; | ||||
|     } | ||||
|     auto opt = 0; | ||||
|     if (ui->authorsCheckBox->isChecked()) | ||||
|         opt |= ExportToModList::Authors; | ||||
|     if (ui->versionCheckBox->isChecked()) | ||||
|         opt |= ExportToModList::Version; | ||||
|     if (ui->urlCheckBox->isChecked()) | ||||
|         opt |= ExportToModList::Url; | ||||
|     auto txt = ExportToModList::exportToModList(m_allMods, format, static_cast<ExportToModList::OptionalData>(opt)); | ||||
|     ui->finalText->setPlainText(txt); | ||||
|     switch (format) { | ||||
|         case ExportToModList::CUSTOM: | ||||
|             return; | ||||
|         case ExportToModList::HTML: | ||||
|             ui->resultText->setHtml(txt); | ||||
|             break; | ||||
|         case ExportToModList::MARKDOWN: | ||||
|             ui->resultText->setHtml(markdownToHTML(txt)); | ||||
|             break; | ||||
|         case ExportToModList::PLAINTXT: | ||||
|             break; | ||||
|         case ExportToModList::JSON: | ||||
|             break; | ||||
|         case ExportToModList::CSV: | ||||
|             break; | ||||
|     } | ||||
|     auto exampleLine = exampleLines[format]; | ||||
|     if (!m_template_changed && ui->templateText->toPlainText() != exampleLine) | ||||
|         ui->templateText->setPlainText(exampleLine); | ||||
| } | ||||
|  | ||||
| void ExportToModListDialog::done(int result) | ||||
| { | ||||
|     if (result == Accepted) { | ||||
|         const QString filename = FS::RemoveInvalidFilenameChars(name); | ||||
|         const QString output = | ||||
|             QFileDialog::getSaveFileName(this, tr("Export %1").arg(name), FS::PathCombine(QDir::homePath(), filename + extension()), | ||||
|                                          "File (*.txt *.html *.md *.json *.csv)", nullptr); | ||||
|  | ||||
|         if (output.isEmpty()) | ||||
|             return; | ||||
|         FS::write(output, ui->finalText->toPlainText().toUtf8()); | ||||
|     } | ||||
|  | ||||
|     QDialog::done(result); | ||||
| } | ||||
|  | ||||
| QString ExportToModListDialog::extension() | ||||
| { | ||||
|     switch (format) { | ||||
|         case ExportToModList::HTML: | ||||
|             return ".html"; | ||||
|         case ExportToModList::MARKDOWN: | ||||
|             return ".md"; | ||||
|         case ExportToModList::PLAINTXT: | ||||
|             return ".txt"; | ||||
|         case ExportToModList::CUSTOM: | ||||
|             return ".txt"; | ||||
|         case ExportToModList::JSON: | ||||
|             return ".json"; | ||||
|         case ExportToModList::CSV: | ||||
|             return ".csv"; | ||||
|     } | ||||
|     return ".txt"; | ||||
| } | ||||
|  | ||||
| void ExportToModListDialog::addExtra(ExportToModList::OptionalData option) | ||||
| { | ||||
|     if (format != ExportToModList::CUSTOM) | ||||
|         return; | ||||
|     switch (option) { | ||||
|         case ExportToModList::Authors: | ||||
|             ui->templateText->insertPlainText("{authors}"); | ||||
|             break; | ||||
|         case ExportToModList::Url: | ||||
|             ui->templateText->insertPlainText("{url}"); | ||||
|             break; | ||||
|         case ExportToModList::Version: | ||||
|             ui->templateText->insertPlainText("{version}"); | ||||
|             break; | ||||
|     } | ||||
| } | ||||
| void ExportToModListDialog::enableCustom(bool enabled) | ||||
| { | ||||
|     ui->authorsCheckBox->setHidden(enabled); | ||||
|     ui->versionCheckBox->setHidden(enabled); | ||||
|     ui->urlCheckBox->setHidden(enabled); | ||||
|  | ||||
|     ui->authorsButton->setHidden(!enabled); | ||||
|     ui->versionButton->setHidden(!enabled); | ||||
|     ui->urlButton->setHidden(!enabled); | ||||
| } | ||||
							
								
								
									
										55
									
								
								launcher/ui/dialogs/ExportToModListDialog.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								launcher/ui/dialogs/ExportToModListDialog.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, version 3. | ||||
|  * | ||||
|  *  This program is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include <QDialog> | ||||
| #include <QList> | ||||
| #include "BaseInstance.h" | ||||
| #include "minecraft/mod/Mod.h" | ||||
| #include "modplatform/helpers/ExportToModList.h" | ||||
|  | ||||
| namespace Ui { | ||||
| class ExportToModListDialog; | ||||
| } | ||||
|  | ||||
| class ExportToModListDialog : public QDialog { | ||||
|     Q_OBJECT | ||||
|  | ||||
|    public: | ||||
|     explicit ExportToModListDialog(InstancePtr instance, QWidget* parent = nullptr); | ||||
|     ~ExportToModListDialog(); | ||||
|  | ||||
|     void done(int result) override; | ||||
|  | ||||
|    protected slots: | ||||
|     void formatChanged(int index); | ||||
|     void triggerImp(); | ||||
|     void trigger(int) { triggerImp(); }; | ||||
|     void addExtra(ExportToModList::OptionalData option); | ||||
|  | ||||
|    private: | ||||
|     QString extension(); | ||||
|     void enableCustom(bool enabled); | ||||
|     QList<Mod*> m_allMods; | ||||
|     bool m_template_changed; | ||||
|     QString name; | ||||
|     ExportToModList::Formats format = ExportToModList::Formats::HTML; | ||||
|     Ui::ExportToModListDialog* ui; | ||||
|     static const QHash<ExportToModList::Formats, QString> exampleLines; | ||||
| }; | ||||
							
								
								
									
										240
									
								
								launcher/ui/dialogs/ExportToModListDialog.ui
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										240
									
								
								launcher/ui/dialogs/ExportToModListDialog.ui
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,240 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <ui version="4.0"> | ||||
|  <class>ExportToModListDialog</class> | ||||
|  <widget class="QDialog" name="ExportToModListDialog"> | ||||
|   <property name="geometry"> | ||||
|    <rect> | ||||
|     <x>0</x> | ||||
|     <y>0</y> | ||||
|     <width>650</width> | ||||
|     <height>446</height> | ||||
|    </rect> | ||||
|   </property> | ||||
|   <property name="windowTitle"> | ||||
|    <string>Export Pack to ModList</string> | ||||
|   </property> | ||||
|   <property name="sizeGripEnabled"> | ||||
|    <bool>true</bool> | ||||
|   </property> | ||||
|   <layout class="QVBoxLayout" name="verticalLayout_2"> | ||||
|    <item> | ||||
|     <layout class="QVBoxLayout" name="verticalLayout" stretch="0,0,0"> | ||||
|      <item> | ||||
|       <widget class="QGroupBox" name="groupBox_3"> | ||||
|        <property name="title"> | ||||
|         <string>Settings</string> | ||||
|        </property> | ||||
|        <layout class="QGridLayout" name="gridLayout"> | ||||
|         <item row="0" column="1"> | ||||
|          <widget class="QComboBox" name="formatComboBox"> | ||||
|           <item> | ||||
|            <property name="text"> | ||||
|             <string>HTML</string> | ||||
|            </property> | ||||
|           </item> | ||||
|           <item> | ||||
|            <property name="text"> | ||||
|             <string>Markdown</string> | ||||
|            </property> | ||||
|           </item> | ||||
|           <item> | ||||
|            <property name="text"> | ||||
|             <string>Plaintext</string> | ||||
|            </property> | ||||
|           </item> | ||||
|           <item> | ||||
|            <property name="text"> | ||||
|             <string>JSON</string> | ||||
|            </property> | ||||
|           </item> | ||||
|           <item> | ||||
|            <property name="text"> | ||||
|             <string>CSV</string> | ||||
|            </property> | ||||
|           </item> | ||||
|           <item> | ||||
|            <property name="text"> | ||||
|             <string>Custom</string> | ||||
|            </property> | ||||
|           </item> | ||||
|          </widget> | ||||
|         </item> | ||||
|         <item row="1" column="0"> | ||||
|          <widget class="QGroupBox" name="templateGroup"> | ||||
|           <property name="title"> | ||||
|            <string>Template</string> | ||||
|           </property> | ||||
|           <layout class="QVBoxLayout" name="verticalLayout_4"> | ||||
|            <item> | ||||
|             <widget class="QTextEdit" name="templateText"/> | ||||
|            </item> | ||||
|           </layout> | ||||
|          </widget> | ||||
|         </item> | ||||
|         <item row="1" column="1"> | ||||
|          <widget class="QGroupBox" name="optionsGroup"> | ||||
|           <property name="title"> | ||||
|            <string>Optional Info</string> | ||||
|           </property> | ||||
|           <layout class="QVBoxLayout" name="verticalLayout_5"> | ||||
|            <item> | ||||
|             <widget class="QCheckBox" name="versionCheckBox"> | ||||
|              <property name="text"> | ||||
|               <string>Version</string> | ||||
|              </property> | ||||
|             </widget> | ||||
|            </item> | ||||
|            <item> | ||||
|             <widget class="QCheckBox" name="authorsCheckBox"> | ||||
|              <property name="text"> | ||||
|               <string>Authors</string> | ||||
|              </property> | ||||
|             </widget> | ||||
|            </item> | ||||
|            <item> | ||||
|             <widget class="QCheckBox" name="urlCheckBox"> | ||||
|              <property name="text"> | ||||
|               <string>URL</string> | ||||
|              </property> | ||||
|             </widget> | ||||
|            </item> | ||||
|            <item> | ||||
|             <widget class="QPushButton" name="versionButton"> | ||||
|              <property name="text"> | ||||
|               <string>Version</string> | ||||
|              </property> | ||||
|             </widget> | ||||
|            </item> | ||||
|            <item> | ||||
|             <widget class="QPushButton" name="authorsButton"> | ||||
|              <property name="text"> | ||||
|               <string>Authors</string> | ||||
|              </property> | ||||
|             </widget> | ||||
|            </item> | ||||
|            <item> | ||||
|             <widget class="QPushButton" name="urlButton"> | ||||
|              <property name="text"> | ||||
|               <string>URL</string> | ||||
|              </property> | ||||
|             </widget> | ||||
|            </item> | ||||
|           </layout> | ||||
|          </widget> | ||||
|         </item> | ||||
|         <item row="0" column="0"> | ||||
|          <widget class="QLabel" name="label"> | ||||
|           <property name="frameShape"> | ||||
|            <enum>QFrame::NoFrame</enum> | ||||
|           </property> | ||||
|           <property name="frameShadow"> | ||||
|            <enum>QFrame::Plain</enum> | ||||
|           </property> | ||||
|           <property name="lineWidth"> | ||||
|            <number>1</number> | ||||
|           </property> | ||||
|           <property name="text"> | ||||
|            <string>Format</string> | ||||
|           </property> | ||||
|          </widget> | ||||
|         </item> | ||||
|        </layout> | ||||
|       </widget> | ||||
|      </item> | ||||
|      <item> | ||||
|       <widget class="QGroupBox" name="groupBox_4"> | ||||
|        <property name="title"> | ||||
|         <string>Result</string> | ||||
|        </property> | ||||
|        <layout class="QHBoxLayout" name="horizontalLayout"> | ||||
|         <item> | ||||
|          <widget class="QPlainTextEdit" name="finalText"> | ||||
|           <property name="minimumSize"> | ||||
|            <size> | ||||
|             <width>0</width> | ||||
|             <height>143</height> | ||||
|            </size> | ||||
|           </property> | ||||
|           <property name="readOnly"> | ||||
|            <bool>true</bool> | ||||
|           </property> | ||||
|          </widget> | ||||
|         </item> | ||||
|         <item> | ||||
|          <widget class="QTextBrowser" name="resultText"> | ||||
|           <property name="openExternalLinks"> | ||||
|            <bool>true</bool> | ||||
|           </property> | ||||
|          </widget> | ||||
|         </item> | ||||
|        </layout> | ||||
|       </widget> | ||||
|      </item> | ||||
|      <item> | ||||
|       <widget class="QLabel" name="warningLabel"> | ||||
|        <property name="text"> | ||||
|         <string>This depends on the mods' metadata. To ensure it is available, run an update on the instance. Installing the updates isn't necessary.</string> | ||||
|        </property> | ||||
|        <property name="wordWrap"> | ||||
|         <bool>true</bool> | ||||
|        </property> | ||||
|       </widget> | ||||
|      </item> | ||||
|     </layout> | ||||
|    </item> | ||||
|    <item> | ||||
|     <layout class="QHBoxLayout" name="horizontalLayout_2"> | ||||
|      <item> | ||||
|       <widget class="QPushButton" name="copyButton"> | ||||
|        <property name="text"> | ||||
|         <string>Copy</string> | ||||
|        </property> | ||||
|       </widget> | ||||
|      </item> | ||||
|      <item> | ||||
|       <widget class="QDialogButtonBox" name="buttonBox"> | ||||
|        <property name="standardButtons"> | ||||
|         <set>QDialogButtonBox::Cancel|QDialogButtonBox::Save</set> | ||||
|        </property> | ||||
|       </widget> | ||||
|      </item> | ||||
|     </layout> | ||||
|    </item> | ||||
|   </layout> | ||||
|  </widget> | ||||
|  <resources/> | ||||
|  <connections> | ||||
|   <connection> | ||||
|    <sender>buttonBox</sender> | ||||
|    <signal>accepted()</signal> | ||||
|    <receiver>ExportToModListDialog</receiver> | ||||
|    <slot>accept()</slot> | ||||
|    <hints> | ||||
|     <hint type="sourcelabel"> | ||||
|      <x>334</x> | ||||
|      <y>435</y> | ||||
|     </hint> | ||||
|     <hint type="destinationlabel"> | ||||
|      <x>324</x> | ||||
|      <y>206</y> | ||||
|     </hint> | ||||
|    </hints> | ||||
|   </connection> | ||||
|   <connection> | ||||
|    <sender>buttonBox</sender> | ||||
|    <signal>rejected()</signal> | ||||
|    <receiver>ExportToModListDialog</receiver> | ||||
|    <slot>reject()</slot> | ||||
|    <hints> | ||||
|     <hint type="sourcelabel"> | ||||
|      <x>324</x> | ||||
|      <y>390</y> | ||||
|     </hint> | ||||
|     <hint type="destinationlabel"> | ||||
|      <x>324</x> | ||||
|      <y>206</y> | ||||
|     </hint> | ||||
|    </hints> | ||||
|   </connection> | ||||
|  </connections> | ||||
| </ui> | ||||
| @@ -1,7 +1,8 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  PolyMC - Minecraft Launcher | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net> | ||||
|  *  Copyright (C) 2023 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
| @@ -35,22 +36,18 @@ | ||||
|  | ||||
| #include "VisualGroup.h" | ||||
|  | ||||
| #include <QApplication> | ||||
| #include <QDebug> | ||||
| #include <QModelIndex> | ||||
| #include <QPainter> | ||||
| #include <QtMath> | ||||
| #include <QApplication> | ||||
| #include <QDebug> | ||||
| #include <utility> | ||||
|  | ||||
| #include "InstanceView.h" | ||||
|  | ||||
| VisualGroup::VisualGroup(const QString &text, InstanceView *view) : view(view), text(text), collapsed(false) | ||||
| { | ||||
| } | ||||
| VisualGroup::VisualGroup(QString text, InstanceView* view) : view(view), text(std::move(text)), collapsed(false) {} | ||||
|  | ||||
| VisualGroup::VisualGroup(const VisualGroup *other) | ||||
|     : view(other->view), text(other->text), collapsed(other->collapsed) | ||||
| { | ||||
| } | ||||
| VisualGroup::VisualGroup(const VisualGroup* other) : view(other->view), text(other->text), collapsed(other->collapsed) {} | ||||
|  | ||||
| void VisualGroup::update() | ||||
| { | ||||
| @@ -64,13 +61,11 @@ void VisualGroup::update() | ||||
|     int positionInRow = 0; | ||||
|     int currentRow = 0; | ||||
|     int offsetFromTop = 0; | ||||
|     for (auto item: temp_items) | ||||
|     { | ||||
|         if(positionInRow == itemsPerRow) | ||||
|         { | ||||
|     for (auto item : temp_items) { | ||||
|         if (positionInRow == itemsPerRow) { | ||||
|             rows[currentRow].height = maxRowHeight; | ||||
|             rows[currentRow].top = offsetFromTop; | ||||
|             currentRow ++; | ||||
|             currentRow++; | ||||
|             offsetFromTop += maxRowHeight + 5; | ||||
|             positionInRow = 0; | ||||
|             maxRowHeight = 0; | ||||
| @@ -83,8 +78,7 @@ void VisualGroup::update() | ||||
| #endif | ||||
|  | ||||
|         auto itemHeight = view->itemDelegate()->sizeHint(viewItemOption, item).height(); | ||||
|         if(itemHeight > maxRowHeight) | ||||
|         { | ||||
|         if (itemHeight > maxRowHeight) { | ||||
|             maxRowHeight = itemHeight; | ||||
|         } | ||||
|         rows[currentRow].items.append(item); | ||||
| @@ -94,16 +88,13 @@ void VisualGroup::update() | ||||
|     rows[currentRow].top = offsetFromTop; | ||||
| } | ||||
|  | ||||
| QPair<int, int> VisualGroup::positionOf(const QModelIndex &index) const | ||||
| QPair<int, int> VisualGroup::positionOf(const QModelIndex& index) const | ||||
| { | ||||
|     int y = 0; | ||||
|     for (auto & row: rows) | ||||
|     { | ||||
|         for(auto x = 0; x < row.items.size(); x++) | ||||
|         { | ||||
|             if(row.items[x] == index) | ||||
|             { | ||||
|                 return qMakePair(x,y); | ||||
|     for (auto& row : rows) { | ||||
|         for (auto x = 0; x < row.items.size(); x++) { | ||||
|             if (row.items[x] == index) { | ||||
|                 return qMakePair(x, y); | ||||
|             } | ||||
|         } | ||||
|         y++; | ||||
| @@ -112,186 +103,102 @@ QPair<int, int> VisualGroup::positionOf(const QModelIndex &index) const | ||||
|     return qMakePair(0, 0); | ||||
| } | ||||
|  | ||||
| int VisualGroup::rowTopOf(const QModelIndex &index) const | ||||
| int VisualGroup::rowTopOf(const QModelIndex& index) const | ||||
| { | ||||
|     auto position = positionOf(index); | ||||
|     return rows[position.second].top; | ||||
| } | ||||
|  | ||||
| int VisualGroup::rowHeightOf(const QModelIndex &index) const | ||||
| int VisualGroup::rowHeightOf(const QModelIndex& index) const | ||||
| { | ||||
|     auto position = positionOf(index); | ||||
|     return rows[position.second].height; | ||||
| } | ||||
|  | ||||
| VisualGroup::HitResults VisualGroup::hitScan(const QPoint &pos) const | ||||
| VisualGroup::HitResults VisualGroup::hitScan(const QPoint& pos) const | ||||
| { | ||||
|     VisualGroup::HitResults results = VisualGroup::NoHit; | ||||
|     int y_start = verticalPosition(); | ||||
|     int body_start = y_start + headerHeight(); | ||||
|     int body_end = body_start + contentHeight() + 5; // FIXME: wtf is this 5? | ||||
|     int body_end = body_start + contentHeight(); | ||||
|     int y = pos.y(); | ||||
|     // int x = pos.x(); | ||||
|     if (y < y_start) | ||||
|     { | ||||
|     if (y < y_start) { | ||||
|         results = VisualGroup::NoHit; | ||||
|     } | ||||
|     else if (y < body_start) | ||||
|     { | ||||
|     } else if (y < body_start) { | ||||
|         results = VisualGroup::HeaderHit; | ||||
|         int collapseSize = headerHeight() - 4; | ||||
|  | ||||
|         // the icon | ||||
|         QRect iconRect = QRect(view->m_leftMargin + 2, 2 + y_start, collapseSize, collapseSize); | ||||
|         if (iconRect.contains(pos)) | ||||
|         { | ||||
|         QRect iconRect = QRect(view->m_leftMargin + 2, 2 + y_start, view->width() - 4, collapseSize); | ||||
|         if (iconRect.contains(pos)) { | ||||
|             results |= VisualGroup::CheckboxHit; | ||||
|         } | ||||
|     } | ||||
|     else if (y < body_end) | ||||
|     { | ||||
|     } else if (y < body_end) { | ||||
|         results |= VisualGroup::BodyHit; | ||||
|     } | ||||
|     return results; | ||||
| } | ||||
|  | ||||
| void VisualGroup::drawHeader(QPainter *painter, const QStyleOptionViewItem &option) | ||||
| void VisualGroup::drawHeader(QPainter* painter, const QStyleOptionViewItem& option) const | ||||
| { | ||||
|     painter->setRenderHint(QPainter::Antialiasing); | ||||
|  | ||||
|     const QRect optRect = option.rect; | ||||
|     QRect optRect = option.rect; | ||||
|     optRect.setTop(optRect.top() + 7); | ||||
|     QFont font(QApplication::font()); | ||||
|     font.setBold(true); | ||||
|     const QFontMetrics fontMetrics = QFontMetrics(font); | ||||
|  | ||||
|     QColor outlineColor = option.palette.text().color(); | ||||
|     outlineColor.setAlphaF(0.35); | ||||
|  | ||||
|     //BEGIN: top left corner | ||||
|     { | ||||
|         painter->save(); | ||||
|         painter->setPen(outlineColor); | ||||
|         const QPointF topLeft(optRect.topLeft()); | ||||
|         QRectF arc(topLeft, QSizeF(4, 4)); | ||||
|         arc.translate(0.5, 0.5); | ||||
|         painter->drawArc(arc, 1440, 1440); | ||||
|         painter->restore(); | ||||
|     } | ||||
|     //END: top left corner | ||||
|  | ||||
|     //BEGIN: left vertical line | ||||
|     { | ||||
|         QPoint start(optRect.topLeft()); | ||||
|         start.ry() += 3; | ||||
|         QPoint verticalGradBottom(optRect.topLeft()); | ||||
|         verticalGradBottom.ry() += fontMetrics.height() + 5; | ||||
|         QLinearGradient gradient(start, verticalGradBottom); | ||||
|         gradient.setColorAt(0, outlineColor); | ||||
|         gradient.setColorAt(1, Qt::transparent); | ||||
|         painter->fillRect(QRect(start, QSize(1, fontMetrics.height() + 5)), gradient); | ||||
|     } | ||||
|     //END: left vertical line | ||||
|  | ||||
|     //BEGIN: horizontal line | ||||
|     { | ||||
|         QPoint start(optRect.topLeft()); | ||||
|         start.rx() += 3; | ||||
|         QPoint horizontalGradTop(optRect.topLeft()); | ||||
|         horizontalGradTop.rx() += optRect.width() - 6; | ||||
|         painter->fillRect(QRect(start, QSize(optRect.width() - 6, 1)), outlineColor); | ||||
|     } | ||||
|     //END: horizontal line | ||||
|  | ||||
|     //BEGIN: top right corner | ||||
|     { | ||||
|         painter->save(); | ||||
|         painter->setPen(outlineColor); | ||||
|         QPointF topRight(optRect.topRight()); | ||||
|         topRight.rx() -= 4; | ||||
|         QRectF arc(topRight, QSizeF(4, 4)); | ||||
|         arc.translate(0.5, 0.5); | ||||
|         painter->drawArc(arc, 0, 1440); | ||||
|         painter->restore(); | ||||
|     } | ||||
|     //END: top right corner | ||||
|  | ||||
|     //BEGIN: right vertical line | ||||
|     { | ||||
|         QPoint start(optRect.topRight()); | ||||
|         start.ry() += 3; | ||||
|         QPoint verticalGradBottom(optRect.topRight()); | ||||
|         verticalGradBottom.ry() += fontMetrics.height() + 5; | ||||
|         QLinearGradient gradient(start, verticalGradBottom); | ||||
|         gradient.setColorAt(0, outlineColor); | ||||
|         gradient.setColorAt(1, Qt::transparent); | ||||
|         painter->fillRect(QRect(start, QSize(1, fontMetrics.height() + 5)), gradient); | ||||
|     } | ||||
|     //END: right vertical line | ||||
|  | ||||
|     //BEGIN: checkboxy thing | ||||
|     { | ||||
|         painter->save(); | ||||
|         painter->setRenderHint(QPainter::Antialiasing, false); | ||||
|     painter->setFont(font); | ||||
|         QColor penColor(option.palette.text().color()); | ||||
|  | ||||
|     QPen pen; | ||||
|     pen.setWidth(2); | ||||
|     QColor penColor = option.palette.text().color(); | ||||
|     penColor.setAlphaF(0.6); | ||||
|         painter->setPen(penColor); | ||||
|         QRect iconSubRect(option.rect); | ||||
|         iconSubRect.setTop(iconSubRect.top() + 7); | ||||
|         iconSubRect.setLeft(iconSubRect.left() + 7); | ||||
|     pen.setColor(penColor); | ||||
|     painter->setPen(pen); | ||||
|     painter->setRenderHint(QPainter::Antialiasing); | ||||
|  | ||||
|         int sizing = fontMetrics.height(); | ||||
|         int even = ( (sizing - 1) % 2 ); | ||||
|     // sizes and offsets, to keep things consistent below | ||||
|     int arrowOffsetLeft = fontMetrics.height() / 2 + 7; | ||||
|     int textOffsetLeft = arrowOffsetLeft * 2; | ||||
|     int arrowSize = 6; | ||||
|     int centerHeight = optRect.top() + fontMetrics.height() / 2; | ||||
|  | ||||
|         iconSubRect.setHeight(sizing - even); | ||||
|         iconSubRect.setWidth(sizing - even); | ||||
|         painter->drawRect(iconSubRect); | ||||
|  | ||||
|  | ||||
|         /* | ||||
|         if(collapsed) | ||||
|             painter->drawText(iconSubRect, Qt::AlignHCenter | Qt::AlignVCenter, "+"); | ||||
|         else | ||||
|             painter->drawText(iconSubRect, Qt::AlignHCenter | Qt::AlignVCenter, "-"); | ||||
|         */ | ||||
|         painter->setBrush(option.palette.text()); | ||||
|         painter->fillRect(iconSubRect.x(), iconSubRect.y() + iconSubRect.height() / 2, | ||||
|                           iconSubRect.width(), 2, penColor); | ||||
|         if (collapsed) | ||||
|     // BEGIN: arrow | ||||
|     { | ||||
|             painter->fillRect(iconSubRect.x() + iconSubRect.width() / 2, iconSubRect.y(), 2, | ||||
|                               iconSubRect.height(), penColor); | ||||
|         QPolygon arrowPolygon; | ||||
|         if (collapsed) { | ||||
|             arrowPolygon << QPoint(arrowOffsetLeft - arrowSize / 2, centerHeight - arrowSize) | ||||
|                          << QPoint(arrowOffsetLeft + arrowSize / 2, centerHeight) | ||||
|                          << QPoint(arrowOffsetLeft - arrowSize / 2, centerHeight + arrowSize); | ||||
|             painter->drawPolyline(arrowPolygon); | ||||
|         } else { | ||||
|             arrowPolygon << QPoint(arrowOffsetLeft - arrowSize, centerHeight - arrowSize / 2) | ||||
|                          << QPoint(arrowOffsetLeft, centerHeight + arrowSize / 2) | ||||
|                          << QPoint(arrowOffsetLeft + arrowSize, centerHeight - arrowSize / 2); | ||||
|             painter->drawPolyline(arrowPolygon); | ||||
|         } | ||||
|  | ||||
|         painter->restore(); | ||||
|     } | ||||
|     //END: checkboxy thing | ||||
|     // END: arrow | ||||
|  | ||||
|     //BEGIN: text | ||||
|     // BEGIN: text | ||||
|     { | ||||
|         QRect textRect(option.rect); | ||||
|         textRect.setTop(textRect.top() + 7); | ||||
|         textRect.setLeft(textRect.left() + 7 + fontMetrics.height() + 7); | ||||
|         QRect textRect(optRect); | ||||
|         textRect.setTop(textRect.top()); | ||||
|         textRect.setLeft(textOffsetLeft); | ||||
|         textRect.setHeight(fontMetrics.height()); | ||||
|         textRect.setRight(textRect.right() - 7); | ||||
|  | ||||
|         painter->save(); | ||||
|         painter->setFont(font); | ||||
|         QColor penColor(option.palette.text().color()); | ||||
|         penColor.setAlphaF(0.6); | ||||
|         painter->setPen(penColor); | ||||
|         painter->drawText(textRect, Qt::AlignLeft | Qt::AlignVCenter, text); | ||||
|         painter->restore(); | ||||
|         painter->drawText(textRect, Qt::AlignLeft | Qt::AlignVCenter, !text.isEmpty() ? text : QObject::tr("Ungrouped")); | ||||
|     } | ||||
|     //END: text | ||||
|     // END: text | ||||
| } | ||||
|  | ||||
| int VisualGroup::totalHeight() const | ||||
| { | ||||
|     return headerHeight() + 5 + contentHeight(); // FIXME: wtf is that '5'? | ||||
|     return headerHeight() + contentHeight(); | ||||
| } | ||||
|  | ||||
| int VisualGroup::headerHeight() const | ||||
| int VisualGroup::headerHeight() | ||||
| { | ||||
|     QFont font(QApplication::font()); | ||||
|     font.setBold(true); | ||||
| @@ -311,8 +218,7 @@ int VisualGroup::headerHeight() const | ||||
|  | ||||
| int VisualGroup::contentHeight() const | ||||
| { | ||||
|     if (collapsed) | ||||
|     { | ||||
|     if (collapsed) { | ||||
|         return 0; | ||||
|     } | ||||
|     auto last = rows[numRows() - 1]; | ||||
| @@ -321,7 +227,7 @@ int VisualGroup::contentHeight() const | ||||
|  | ||||
| int VisualGroup::numRows() const | ||||
| { | ||||
|     return rows.size(); | ||||
|     return (int)rows.size(); | ||||
| } | ||||
|  | ||||
| int VisualGroup::verticalPosition() const | ||||
| @@ -332,11 +238,9 @@ int VisualGroup::verticalPosition() const | ||||
| QList<QModelIndex> VisualGroup::items() const | ||||
| { | ||||
|     QList<QModelIndex> indices; | ||||
|     for (int i = 0; i < view->model()->rowCount(); ++i) | ||||
|     { | ||||
|     for (int i = 0; i < view->model()->rowCount(); ++i) { | ||||
|         const QModelIndex index = view->model()->index(i, 0); | ||||
|         if (index.data(InstanceViewRoles::GroupRole).toString() == text) | ||||
|         { | ||||
|         if (index.data(InstanceViewRoles::GroupRole).toString() == text) { | ||||
|             indices.append(index); | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -1,4 +1,24 @@ | ||||
| /* Copyright 2013-2021 MultiMC Contributors | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2023 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|  *  the Free Software Foundation, version 3. | ||||
|  * | ||||
|  *  This program is distributed in the hope that it will be useful, | ||||
|  *  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  *  GNU General Public License for more details. | ||||
|  * | ||||
|  *  You should have received a copy of the GNU General Public License | ||||
|  *  along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  * | ||||
|  * This file incorporates work covered by the following copyright and | ||||
|  * permission notice: | ||||
|  * | ||||
|  *      Copyright 2013-2021 MultiMC Contributors | ||||
|  * | ||||
|  *      Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  *      you may not use this file except in compliance with the License. | ||||
| @@ -42,8 +62,8 @@ struct VisualRow | ||||
| struct VisualGroup | ||||
| { | ||||
| /* constructors */ | ||||
|     VisualGroup(const QString &text, InstanceView *view); | ||||
|     VisualGroup(const VisualGroup *other); | ||||
|     VisualGroup(QString text, InstanceView *view); | ||||
|     explicit VisualGroup(const VisualGroup *other); | ||||
|  | ||||
| /* data */ | ||||
|     InstanceView *view = nullptr; | ||||
| @@ -58,13 +78,13 @@ struct VisualGroup | ||||
|     void update(); | ||||
|  | ||||
|     /// draw the header at y-position. | ||||
|     void drawHeader(QPainter *painter, const QStyleOptionViewItem &option); | ||||
|     void drawHeader(QPainter *painter, const QStyleOptionViewItem &option) const; | ||||
|  | ||||
|     /// height of the group, in total. includes a small bit of padding. | ||||
|     int totalHeight() const; | ||||
|  | ||||
|     /// height of the group header, in pixels | ||||
|     int headerHeight() const; | ||||
|     static int headerHeight() ; | ||||
|  | ||||
|     /// height of the group content, in pixels | ||||
|     int contentHeight() const; | ||||
|   | ||||
| @@ -159,19 +159,6 @@ void AccountListPage::on_actionAddMojang_triggered() | ||||
|  | ||||
| void AccountListPage::on_actionAddMicrosoft_triggered() | ||||
| { | ||||
|     if(BuildConfig.BUILD_PLATFORM == "osx64") { | ||||
|         CustomMessageBox::selectable( | ||||
|             this, | ||||
|             tr("Microsoft Accounts not available"), | ||||
|             //: %1 refers to the launcher itself | ||||
|             tr( | ||||
|                 "Microsoft accounts are only usable on macOS 10.13 or newer, with fully updated %1.\n\n" | ||||
|                 "Please update both your operating system and %1." | ||||
|             ).arg(BuildConfig.LAUNCHER_DISPLAYNAME), | ||||
|             QMessageBox::Warning | ||||
|         )->exec(); | ||||
|         return; | ||||
|     } | ||||
|     MinecraftAccountPtr account = MSALoginDialog::newAccount( | ||||
|         this, | ||||
|         tr("Please enter your Mojang account email and password to add your account.") | ||||
|   | ||||
| @@ -3,7 +3,7 @@ | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org> | ||||
|  *  Copyright (c) 2022 dada513 <dada513@protonmail.com> | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
| /* | ||||
|  *  PolyMC - Minecraft Launcher | ||||
|  *  Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org> | ||||
|  *  Copyright (C) 2023 seth <getchoo at tuta dot io> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
| @@ -99,6 +100,9 @@ void MinecraftPage::applySettings() | ||||
|     // Miscellaneous | ||||
|     s->set("CloseAfterLaunch", ui->closeAfterLaunchCheck->isChecked()); | ||||
|     s->set("QuitAfterGameStop", ui->quitAfterGameStopCheck->isChecked()); | ||||
|  | ||||
|     // Mod loader settings | ||||
|     s->set("DisableQuiltBeacon", ui->disableQuiltBeaconCheckBox->isChecked()); | ||||
| } | ||||
|  | ||||
| void MinecraftPage::loadSettings() | ||||
| @@ -137,6 +141,8 @@ void MinecraftPage::loadSettings() | ||||
|  | ||||
|     ui->closeAfterLaunchCheck->setChecked(s->get("CloseAfterLaunch").toBool()); | ||||
|     ui->quitAfterGameStopCheck->setChecked(s->get("QuitAfterGameStop").toBool()); | ||||
|  | ||||
|     ui->disableQuiltBeaconCheckBox->setChecked(s->get("DisableQuiltBeacon").toBool()); | ||||
| } | ||||
|  | ||||
| void MinecraftPage::retranslate() | ||||
|   | ||||
| @@ -190,6 +190,25 @@ | ||||
|        <string>Tweaks</string> | ||||
|       </attribute> | ||||
|       <layout class="QVBoxLayout" name="verticalLayout_12"> | ||||
|        <item> | ||||
|         <widget class="QGroupBox" name="modLoaderSettingsGroupBox"> | ||||
|          <property name="title"> | ||||
|           <string>Mod loader settings</string> | ||||
|          </property> | ||||
|          <layout class="QVBoxLayout" name="verticalLayout_13"> | ||||
|           <item> | ||||
|            <widget class="QCheckBox" name="disableQuiltBeaconCheckBox"> | ||||
|             <property name="text"> | ||||
|              <string>Disable Quilt Loader Beacon</string> | ||||
|             </property> | ||||
|             <property name="toolTip"> | ||||
|              <string>Disable Quilt loader's beacon for counting monthly active users</string> | ||||
|             </property> | ||||
|            </widget> | ||||
|           </item> | ||||
|          </layout> | ||||
|         </widget> | ||||
|        </item> | ||||
|        <item> | ||||
|         <widget class="QGroupBox" name="nativeLibWorkaroundGroupBox"> | ||||
|          <property name="title"> | ||||
|   | ||||
| @@ -3,6 +3,7 @@ | ||||
|  *  PolyMC - Minecraft Launcher | ||||
|  *  Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org> | ||||
|  *  Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net> | ||||
|  *  Copyright (C) 2023 seth <getchoo at tuta dot io> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
| @@ -50,9 +51,9 @@ | ||||
| #include "Application.h" | ||||
| #include "minecraft/auth/AccountList.h" | ||||
|  | ||||
| #include "FileSystem.h" | ||||
| #include "java/JavaInstallList.h" | ||||
| #include "java/JavaUtils.h" | ||||
| #include "FileSystem.h" | ||||
|  | ||||
| InstanceSettingsPage::InstanceSettingsPage(BaseInstance *inst, QWidget *parent) | ||||
|     : QWidget(parent), ui(new Ui::InstanceSettingsPage), m_instance(inst) | ||||
| @@ -280,6 +281,14 @@ void InstanceSettingsPage::applySettings() | ||||
|         m_settings->reset("InstanceAccountId"); | ||||
|     } | ||||
|  | ||||
|     bool overrideModLoaderSettings = ui->modLoaderSettingsGroupBox->isChecked(); | ||||
|     m_settings->set("OverrideModLoaderSettings", overrideModLoaderSettings); | ||||
|     if (overrideModLoaderSettings) { | ||||
|         m_settings->set("DisableQuiltBeacon", ui->disableQuiltBeaconCheckBox->isChecked()); | ||||
|     } else { | ||||
|         m_settings->reset("DisableQuiltBeacon"); | ||||
|     } | ||||
|  | ||||
|     // FIXME: This should probably be called by a signal instead | ||||
|     m_instance->updateRuntimeContext(); | ||||
| } | ||||
| @@ -380,6 +389,10 @@ void InstanceSettingsPage::loadSettings() | ||||
|  | ||||
|     ui->instanceAccountGroupBox->setChecked(m_settings->get("UseAccountForInstance").toBool()); | ||||
|     updateAccountsMenu(); | ||||
|  | ||||
|     // Mod loader specific settings | ||||
|     ui->modLoaderSettingsGroupBox->setChecked(m_settings->get("OverrideModLoaderSettings").toBool()); | ||||
|     ui->disableQuiltBeaconCheckBox->setChecked(m_settings->get("DisableQuiltBeacon").toBool()); | ||||
| } | ||||
|  | ||||
| void InstanceSettingsPage::on_javaDetectBtn_clicked() | ||||
|   | ||||
| @@ -541,6 +541,31 @@ | ||||
|        <string>Miscellaneous</string> | ||||
|       </attribute> | ||||
|       <layout class="QVBoxLayout" name="verticalLayout_9"> | ||||
|        <item> | ||||
|         <widget class="QGroupBox" name="modLoaderSettingsGroupBox"> | ||||
|          <property name="checkable"> | ||||
|           <bool>true</bool> | ||||
|          </property> | ||||
|          <property name="checked"> | ||||
|           <bool>false</bool> | ||||
|          </property> | ||||
|          <property name="title"> | ||||
|           <string>Mod loader settings</string> | ||||
|          </property> | ||||
|          <layout class="QVBoxLayout" name="VerticalLayout_16"> | ||||
|           <item> | ||||
|            <widget class="QCheckBox" name="disableQuiltBeaconCheckBox"> | ||||
|             <property name="text"> | ||||
|              <string>Disable Quilt Loader Beacon</string> | ||||
|             </property> | ||||
|             <property name="toolTip"> | ||||
|              <string>Disable Quilt loader's beacon for counting monthly active users</string> | ||||
|             </property> | ||||
|            </widget> | ||||
|           </item> | ||||
|          </layout> | ||||
|         </widget> | ||||
|        </item> | ||||
|        <item> | ||||
|         <widget class="QGroupBox" name="gameTimeGroupBox"> | ||||
|          <property name="enabled"> | ||||
|   | ||||
| @@ -104,6 +104,7 @@ void ResourcePage::openedImpl() | ||||
|  | ||||
|     updateSelectionButton(); | ||||
|     triggerSearch(); | ||||
|     m_ui->searchEdit->setFocus(); | ||||
| } | ||||
|  | ||||
| auto ResourcePage::eventFilter(QObject* watched, QEvent* event) -> bool | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,16 +1,16 @@ | ||||
| #include "LanguageSelectionWidget.h" | ||||
|  | ||||
| #include <QVBoxLayout> | ||||
| #include <QTreeView> | ||||
| #include <QCheckBox> | ||||
| #include <QHeaderView> | ||||
| #include <QLabel> | ||||
| #include <QTreeView> | ||||
| #include <QVBoxLayout> | ||||
| #include "Application.h" | ||||
| #include "BuildConfig.h" | ||||
| #include "translations/TranslationsModel.h" | ||||
| #include "settings/Setting.h" | ||||
| #include "translations/TranslationsModel.h" | ||||
|  | ||||
| LanguageSelectionWidget::LanguageSelectionWidget(QWidget *parent) : | ||||
|     QWidget(parent) | ||||
| LanguageSelectionWidget::LanguageSelectionWidget(QWidget* parent) : QWidget(parent) | ||||
| { | ||||
|     verticalLayout = new QVBoxLayout(this); | ||||
|     verticalLayout->setObjectName(QStringLiteral("verticalLayout")); | ||||
| @@ -31,6 +31,13 @@ LanguageSelectionWidget::LanguageSelectionWidget(QWidget *parent) : | ||||
|     helpUsLabel->setWordWrap(true); | ||||
|     verticalLayout->addWidget(helpUsLabel); | ||||
|  | ||||
|     formatCheckbox = new QCheckBox(this); | ||||
|     formatCheckbox->setObjectName(QStringLiteral("formatCheckbox")); | ||||
|     formatCheckbox->setCheckState(APPLICATION->settings()->get("UseSystemLocale").toBool() ? Qt::Checked : Qt::Unchecked); | ||||
|     connect(formatCheckbox, &QCheckBox::stateChanged, | ||||
|             [this]() { APPLICATION->translations()->setUseSystemLocale(formatCheckbox->isChecked()); }); | ||||
|     verticalLayout->addWidget(formatCheckbox); | ||||
|  | ||||
|     auto translations = APPLICATION->translations(); | ||||
|     auto index = translations->selectedIndex(); | ||||
|     languageView->setModel(translations.get()); | ||||
| @@ -38,7 +45,7 @@ LanguageSelectionWidget::LanguageSelectionWidget(QWidget *parent) : | ||||
|     languageView->header()->setSectionResizeMode(QHeaderView::ResizeToContents); | ||||
|     languageView->header()->setSectionResizeMode(0, QHeaderView::Stretch); | ||||
|     connect(languageView->selectionModel(), &QItemSelectionModel::currentRowChanged, this, &LanguageSelectionWidget::languageRowChanged); | ||||
|     verticalLayout->setContentsMargins(0,0,0,0); | ||||
|     verticalLayout->setContentsMargins(0, 0, 0, 0); | ||||
|  | ||||
|     auto language_setting = APPLICATION->settings()->getSetting("Language"); | ||||
|     connect(language_setting.get(), &Setting::SettingChanged, this, &LanguageSelectionWidget::languageSettingChanged); | ||||
| @@ -55,13 +62,12 @@ void LanguageSelectionWidget::retranslate() | ||||
|     QString text = tr("Don't see your language or the quality is poor?<br/><a href=\"%1\">Help us with translations!</a>") | ||||
|                        .arg(BuildConfig.TRANSLATIONS_URL); | ||||
|     helpUsLabel->setText(text); | ||||
|  | ||||
|     formatCheckbox->setText(tr("Use system locales")); | ||||
| } | ||||
|  | ||||
| void LanguageSelectionWidget::languageRowChanged(const QModelIndex& current, const QModelIndex& previous) | ||||
| { | ||||
|     if (current == previous) | ||||
|     { | ||||
|     if (current == previous) { | ||||
|         return; | ||||
|     } | ||||
|     auto translations = APPLICATION->translations(); | ||||
| @@ -70,7 +76,7 @@ void LanguageSelectionWidget::languageRowChanged(const QModelIndex& current, con | ||||
|     translations->updateLanguage(key); | ||||
| } | ||||
|  | ||||
| void LanguageSelectionWidget::languageSettingChanged(const Setting &, const QVariant) | ||||
| void LanguageSelectionWidget::languageSettingChanged(const Setting&, const QVariant) | ||||
| { | ||||
|     auto translations = APPLICATION->translations(); | ||||
|     auto index = translations->selectedIndex(); | ||||
|   | ||||
| @@ -21,23 +21,24 @@ class QVBoxLayout; | ||||
| class QTreeView; | ||||
| class QLabel; | ||||
| class Setting; | ||||
| class QCheckBox; | ||||
|  | ||||
| class LanguageSelectionWidget: public QWidget | ||||
| { | ||||
| class LanguageSelectionWidget : public QWidget { | ||||
|     Q_OBJECT | ||||
| public: | ||||
|     explicit LanguageSelectionWidget(QWidget *parent = 0); | ||||
|     virtual ~LanguageSelectionWidget() { }; | ||||
|    public: | ||||
|     explicit LanguageSelectionWidget(QWidget* parent = 0); | ||||
|     virtual ~LanguageSelectionWidget(){}; | ||||
|  | ||||
|     QString getSelectedLanguageKey() const; | ||||
|     void retranslate(); | ||||
|  | ||||
| protected slots: | ||||
|     void languageRowChanged(const QModelIndex ¤t, const QModelIndex &previous); | ||||
|     void languageSettingChanged(const Setting &, const QVariant); | ||||
|    protected slots: | ||||
|     void languageRowChanged(const QModelIndex& current, const QModelIndex& previous); | ||||
|     void languageSettingChanged(const Setting&, const QVariant); | ||||
|  | ||||
| private: | ||||
|     QVBoxLayout *verticalLayout = nullptr; | ||||
|     QTreeView *languageView = nullptr; | ||||
|     QLabel *helpUsLabel = nullptr; | ||||
|    private: | ||||
|     QVBoxLayout* verticalLayout = nullptr; | ||||
|     QTreeView* languageView = nullptr; | ||||
|     QLabel* helpUsLabel = nullptr; | ||||
|     QCheckBox* formatCheckbox = nullptr; | ||||
| }; | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (C) 2022 Tayou <tayou@gmx.net> | ||||
|  *  Copyright (C) 2022 Tayou <git@tayou.org> | ||||
|  * | ||||
|  *  This program is free software: you can redistribute it and/or modify | ||||
|  *  it under the terms of the GNU General Public License as published by | ||||
|   | ||||
| @@ -42,6 +42,10 @@ class LinkTask : public Task { | ||||
|         m_lnk->debug(true); | ||||
|     } | ||||
|  | ||||
|     ~LinkTask() { | ||||
|         delete m_lnk; | ||||
|     } | ||||
|  | ||||
|     void matcher(const IPathMatcher *filter) | ||||
|     { | ||||
|         m_lnk->matcher(filter); | ||||
| @@ -219,7 +223,8 @@ slots: | ||||
|             qDebug() << tempDir.path(); | ||||
|             qDebug() << target_dir.path(); | ||||
|             FS::copy c(folder, target_dir.path()); | ||||
|             c.matcher(new RegexpMatcher("[.]?mcmeta")); | ||||
|             RegexpMatcher re("[.]?mcmeta"); | ||||
|             c.matcher(&re); | ||||
|             c(); | ||||
|  | ||||
|             for(auto entry: target_dir.entryList()) | ||||
| @@ -253,7 +258,8 @@ slots: | ||||
|             qDebug() << tempDir.path(); | ||||
|             qDebug() << target_dir.path(); | ||||
|             FS::copy c(folder, target_dir.path()); | ||||
|             c.matcher(new RegexpMatcher("[.]?mcmeta")); | ||||
|             RegexpMatcher re("[.]?mcmeta"); | ||||
|             c.matcher(&re); | ||||
|             c.whitelist(true); | ||||
|             c(); | ||||
|  | ||||
| @@ -460,7 +466,8 @@ slots: | ||||
|             qDebug() << target_dir.path(); | ||||
|  | ||||
|             LinkTask lnk_tsk(folder, target_dir.path()); | ||||
|             lnk_tsk.matcher(new RegexpMatcher("[.]?mcmeta")); | ||||
|             RegexpMatcher re("[.]?mcmeta"); | ||||
|             lnk_tsk.matcher(&re); | ||||
|             lnk_tsk.linkRecursively(true); | ||||
|             QObject::connect(&lnk_tsk, &Task::finished, [&]{  | ||||
|                 QVERIFY2(lnk_tsk.wasSuccessful(), "Task finished but was not successful when it should have been.");  | ||||
| @@ -511,7 +518,8 @@ slots: | ||||
|             qDebug() << target_dir.path(); | ||||
|  | ||||
|             LinkTask lnk_tsk(folder, target_dir.path()); | ||||
|             lnk_tsk.matcher(new RegexpMatcher("[.]?mcmeta")); | ||||
|             RegexpMatcher re("[.]?mcmeta"); | ||||
|             lnk_tsk.matcher(&re); | ||||
|             lnk_tsk.linkRecursively(true); | ||||
|             lnk_tsk.whitelist(true); | ||||
|             QObject::connect(&lnk_tsk, &Task::finished, [&]{  | ||||
|   | ||||
| @@ -38,6 +38,7 @@ class DummyResourceModel : public ResourceModel { | ||||
|  | ||||
|    public: | ||||
|     DummyResourceModel() : ResourceModel(new DummyResourceAPI) {} | ||||
|     ~DummyResourceModel() {} | ||||
|  | ||||
|     [[nodiscard]] auto metaEntryBase() const -> QString override { return ""; }; | ||||
|  | ||||
| @@ -58,7 +59,10 @@ class DummyResourceModel : public ResourceModel { | ||||
| class ResourceModelTest : public QObject { | ||||
|     Q_OBJECT | ||||
|    private slots: | ||||
|     void test_abstract_item_model() { [[maybe_unused]] auto tester = new QAbstractItemModelTester(new DummyResourceModel); } | ||||
|     void test_abstract_item_model() {  | ||||
|         auto dummy = DummyResourceModel(); | ||||
|         auto tester = QAbstractItemModelTester(&dummy); | ||||
|     } | ||||
|  | ||||
|     void test_search() | ||||
|     { | ||||
| @@ -78,6 +82,8 @@ class ResourceModelTest : public QObject { | ||||
|         QVERIFY(processed_pack->addonId.toString() == Json::requireString(processed_response, "project_id")); | ||||
|         QVERIFY(processed_pack->description == Json::requireString(processed_response, "description")); | ||||
|         QVERIFY(processed_pack->authors.first().name == Json::requireString(processed_response, "author")); | ||||
|  | ||||
|         delete model; | ||||
|     } | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| #include <QTest> | ||||
| #include <QTimer> | ||||
| #include <QThread> | ||||
| #include <QTimer> | ||||
|  | ||||
| #include <tasks/ConcurrentTask.h> | ||||
| #include <tasks/MultipleOptionsTask.h> | ||||
| @@ -19,10 +19,7 @@ class BasicTask : public Task { | ||||
|     BasicTask(bool show_debug_log = true) : Task(nullptr, show_debug_log) {} | ||||
|  | ||||
|    private: | ||||
|     void executeTask() override | ||||
|     { | ||||
|         emitSucceeded(); | ||||
|     }; | ||||
|     void executeTask() override { emitSucceeded(); }; | ||||
| }; | ||||
|  | ||||
| /* Does nothing. Only used for testing. */ | ||||
| @@ -34,7 +31,7 @@ class BasicTask_MultiStep : public Task { | ||||
|    private: | ||||
|     auto isMultiStep() const -> bool override { return true; } | ||||
|  | ||||
|     void executeTask() override {};    | ||||
|     void executeTask() override{}; | ||||
| }; | ||||
|  | ||||
| class BigConcurrentTask : public ConcurrentTask { | ||||
| @@ -44,7 +41,7 @@ class BigConcurrentTask : public ConcurrentTask { | ||||
|     { | ||||
|         // This is here only to help fill the stack a bit more quickly (if there's an issue, of course :^)) | ||||
|         // Each tasks thus adds 1024 * 4 bytes to the stack, at the very least. | ||||
|         [[maybe_unused]] volatile std::array<uint32_t, 1024> some_data_on_the_stack {}; | ||||
|         [[maybe_unused]] volatile std::array<uint32_t, 1024> some_data_on_the_stack{}; | ||||
|  | ||||
|         ConcurrentTask::startNext(); | ||||
|     } | ||||
| @@ -53,49 +50,42 @@ class BigConcurrentTask : public ConcurrentTask { | ||||
| class BigConcurrentTaskThread : public QThread { | ||||
|     Q_OBJECT | ||||
|  | ||||
|     BigConcurrentTask big_task; | ||||
|  | ||||
|     QTimer m_deadline; | ||||
|     void run() override | ||||
|     { | ||||
|         QTimer deadline; | ||||
|         deadline.setInterval(10000); | ||||
|         connect(&deadline, &QTimer::timeout, this, [this]{ passed_the_deadline = true; }); | ||||
|         deadline.start(); | ||||
|         BigConcurrentTask big_task; | ||||
|         m_deadline.setInterval(10000); | ||||
|  | ||||
|         // NOTE: Arbitrary value that manages to trigger a problem when there is one. | ||||
|         //       Considering each tasks, in a problematic state, adds 1024 * 4 bytes to the stack, | ||||
|         //       this number is enough to fill up 16 MiB of stack, more than enough to cause a problem. | ||||
|         static const unsigned s_num_tasks = 1 << 12; | ||||
|         auto sub_tasks = new BasicTask::Ptr[s_num_tasks]; | ||||
|  | ||||
|         for (unsigned i = 0; i < s_num_tasks; i++) { | ||||
|             auto sub_task = makeShared<BasicTask>(false); | ||||
|             sub_tasks[i] = sub_task; | ||||
|             big_task.addTask(sub_task); | ||||
|         } | ||||
|  | ||||
|         connect(&big_task, &Task::finished, this, &QThread::quit); | ||||
|         connect(&m_deadline, &QTimer::timeout, this, [&] { passed_the_deadline = true; quit(); }); | ||||
|  | ||||
|         m_deadline.start(); | ||||
|         big_task.run(); | ||||
|  | ||||
|         while (!big_task.isFinished() && !passed_the_deadline) | ||||
|             QCoreApplication::processEvents(); | ||||
|  | ||||
|         emit finished(); | ||||
|         exec(); | ||||
|     } | ||||
|  | ||||
|    public: | ||||
|     bool passed_the_deadline = false; | ||||
|  | ||||
|    signals: | ||||
|     void finished(); | ||||
| }; | ||||
|  | ||||
| class TaskTest : public QObject { | ||||
|     Q_OBJECT | ||||
|  | ||||
|    private slots: | ||||
|     void test_SetStatus_NoMultiStep(){ | ||||
|     void test_SetStatus_NoMultiStep() | ||||
|     { | ||||
|         BasicTask t; | ||||
|         QString status {"test status"}; | ||||
|         QString status{ "test status" }; | ||||
|  | ||||
|         t.setStatus(status); | ||||
|  | ||||
| @@ -103,9 +93,10 @@ class TaskTest : public QObject { | ||||
|         QCOMPARE(t.getStepProgress().isEmpty(), TaskStepProgressList{}.isEmpty()); | ||||
|     } | ||||
|  | ||||
|     void test_SetStatus_MultiStep(){ | ||||
|     void test_SetStatus_MultiStep() | ||||
|     { | ||||
|         BasicTask_MultiStep t; | ||||
|         QString status {"test status"}; | ||||
|         QString status{ "test status" }; | ||||
|  | ||||
|         t.setStatus(status); | ||||
|  | ||||
| @@ -115,7 +106,8 @@ class TaskTest : public QObject { | ||||
|         QCOMPARE(t.getStepProgress().isEmpty(), TaskStepProgressList{}.isEmpty()); | ||||
|     } | ||||
|  | ||||
|     void test_SetProgress(){ | ||||
|     void test_SetProgress() | ||||
|     { | ||||
|         BasicTask t; | ||||
|         int current = 42; | ||||
|         int total = 207; | ||||
| @@ -126,17 +118,18 @@ class TaskTest : public QObject { | ||||
|         QCOMPARE(t.getTotalProgress(), total); | ||||
|     } | ||||
|  | ||||
|     void test_basicRun(){ | ||||
|     void test_basicRun() | ||||
|     { | ||||
|         BasicTask t; | ||||
|         QObject::connect(&t, &Task::finished, [&]{ QVERIFY2(t.wasSuccessful(), "Task finished but was not successful when it should have been."); }); | ||||
|         QObject::connect(&t, &Task::finished, | ||||
|                          [&] { QVERIFY2(t.wasSuccessful(), "Task finished but was not successful when it should have been."); }); | ||||
|         t.start(); | ||||
|  | ||||
|         QVERIFY2(QTest::qWaitFor([&]() { | ||||
|             return t.isFinished(); | ||||
|         }, 1000), "Task didn't finish as it should."); | ||||
|         QVERIFY2(QTest::qWaitFor([&]() { return t.isFinished(); }, 1000), "Task didn't finish as it should."); | ||||
|     } | ||||
|  | ||||
|     void test_basicConcurrentRun(){ | ||||
|     void test_basicConcurrentRun() | ||||
|     { | ||||
|         auto t1 = makeShared<BasicTask>(); | ||||
|         auto t2 = makeShared<BasicTask>(); | ||||
|         auto t3 = makeShared<BasicTask>(); | ||||
| @@ -147,7 +140,7 @@ class TaskTest : public QObject { | ||||
|         t.addTask(t2); | ||||
|         t.addTask(t3); | ||||
|  | ||||
|         QObject::connect(&t, &Task::finished, [&]{ | ||||
|         QObject::connect(&t, &Task::finished, [&t, &t1, &t2, &t3] { | ||||
|             QVERIFY2(t.wasSuccessful(), "Task finished but was not successful when it should have been."); | ||||
|             QVERIFY(t1->wasSuccessful()); | ||||
|             QVERIFY(t2->wasSuccessful()); | ||||
| @@ -155,13 +148,12 @@ class TaskTest : public QObject { | ||||
|         }); | ||||
|  | ||||
|         t.start(); | ||||
|         QVERIFY2(QTest::qWaitFor([&]() { | ||||
|             return t.isFinished(); | ||||
|         }, 1000), "Task didn't finish as it should."); | ||||
|         QVERIFY2(QTest::qWaitFor([&]() { return t.isFinished(); }, 1000), "Task didn't finish as it should."); | ||||
|     } | ||||
|  | ||||
|     // Tests if starting new tasks after the 6 initial ones is working | ||||
|     void test_moreConcurrentRun(){ | ||||
|     void test_moreConcurrentRun() | ||||
|     { | ||||
|         auto t1 = makeShared<BasicTask>(); | ||||
|         auto t2 = makeShared<BasicTask>(); | ||||
|         auto t3 = makeShared<BasicTask>(); | ||||
| @@ -184,7 +176,7 @@ class TaskTest : public QObject { | ||||
|         t.addTask(t8); | ||||
|         t.addTask(t9); | ||||
|  | ||||
|         QObject::connect(&t, &Task::finished, [&]{ | ||||
|         QObject::connect(&t, &Task::finished, [&t, &t1, &t2, &t3, &t4, &t5, &t6, &t7, &t8, &t9] { | ||||
|             QVERIFY2(t.wasSuccessful(), "Task finished but was not successful when it should have been."); | ||||
|             QVERIFY(t1->wasSuccessful()); | ||||
|             QVERIFY(t2->wasSuccessful()); | ||||
| @@ -198,12 +190,11 @@ class TaskTest : public QObject { | ||||
|         }); | ||||
|  | ||||
|         t.start(); | ||||
|         QVERIFY2(QTest::qWaitFor([&]() { | ||||
|             return t.isFinished(); | ||||
|         }, 1000), "Task didn't finish as it should."); | ||||
|         QVERIFY2(QTest::qWaitFor([&]() { return t.isFinished(); }, 1000), "Task didn't finish as it should."); | ||||
|     } | ||||
|  | ||||
|     void test_basicSequentialRun(){ | ||||
|     void test_basicSequentialRun() | ||||
|     { | ||||
|         auto t1 = makeShared<BasicTask>(); | ||||
|         auto t2 = makeShared<BasicTask>(); | ||||
|         auto t3 = makeShared<BasicTask>(); | ||||
| @@ -214,7 +205,7 @@ class TaskTest : public QObject { | ||||
|         t.addTask(t2); | ||||
|         t.addTask(t3); | ||||
|  | ||||
|         QObject::connect(&t, &Task::finished, [&]{ | ||||
|         QObject::connect(&t, &Task::finished, [&t, &t1, &t2, &t3] { | ||||
|             QVERIFY2(t.wasSuccessful(), "Task finished but was not successful when it should have been."); | ||||
|             QVERIFY(t1->wasSuccessful()); | ||||
|             QVERIFY(t2->wasSuccessful()); | ||||
| @@ -222,12 +213,11 @@ class TaskTest : public QObject { | ||||
|         }); | ||||
|  | ||||
|         t.start(); | ||||
|         QVERIFY2(QTest::qWaitFor([&]() { | ||||
|             return t.isFinished(); | ||||
|         }, 1000), "Task didn't finish as it should."); | ||||
|         QVERIFY2(QTest::qWaitFor([&]() { return t.isFinished(); }, 1000), "Task didn't finish as it should."); | ||||
|     } | ||||
|  | ||||
|     void test_basicMultipleOptionsRun(){ | ||||
|     void test_basicMultipleOptionsRun() | ||||
|     { | ||||
|         auto t1 = makeShared<BasicTask>(); | ||||
|         auto t2 = makeShared<BasicTask>(); | ||||
|         auto t3 = makeShared<BasicTask>(); | ||||
| @@ -238,7 +228,7 @@ class TaskTest : public QObject { | ||||
|         t.addTask(t2); | ||||
|         t.addTask(t3); | ||||
|  | ||||
|         QObject::connect(&t, &Task::finished, [&]{ | ||||
|         QObject::connect(&t, &Task::finished, [&t, &t1, &t2, &t3] { | ||||
|             QVERIFY2(t.wasSuccessful(), "Task finished but was not successful when it should have been."); | ||||
|             QVERIFY(t1->wasSuccessful()); | ||||
|             QVERIFY(!t2->wasSuccessful()); | ||||
| @@ -246,25 +236,22 @@ class TaskTest : public QObject { | ||||
|         }); | ||||
|  | ||||
|         t.start(); | ||||
|         QVERIFY2(QTest::qWaitFor([&]() { | ||||
|             return t.isFinished(); | ||||
|         }, 1000), "Task didn't finish as it should."); | ||||
|         QVERIFY2(QTest::qWaitFor([&]() { return t.isFinished(); }, 1000), "Task didn't finish as it should."); | ||||
|     } | ||||
|  | ||||
|     void test_stackOverflowInConcurrentTask() | ||||
|     { | ||||
|         QEventLoop loop; | ||||
|  | ||||
|         auto thread = new BigConcurrentTaskThread; | ||||
|         BigConcurrentTaskThread thread; | ||||
|  | ||||
|         connect(thread, &BigConcurrentTaskThread::finished, &loop, &QEventLoop::quit); | ||||
|         connect(&thread, &BigConcurrentTaskThread::finished, &loop, &QEventLoop::quit); | ||||
|  | ||||
|         thread->start(); | ||||
|         thread.start(); | ||||
|  | ||||
|         loop.exec(); | ||||
|  | ||||
|         QVERIFY(!thread->passed_the_deadline); | ||||
|         thread->deleteLater(); | ||||
|         QVERIFY(!thread.passed_the_deadline); | ||||
|     } | ||||
| }; | ||||
|  | ||||
|   | ||||
| @@ -20,6 +20,8 @@ | ||||
| class VersionTest : public QObject { | ||||
|     Q_OBJECT | ||||
|  | ||||
|     QStringList m_flex_test_names = {}; | ||||
|  | ||||
|     void addDataColumns() | ||||
|     { | ||||
|         QTest::addColumn<QString>("first"); | ||||
| @@ -101,8 +103,9 @@ class VersionTest : public QObject { | ||||
|                 QString first{split_line.first().simplified()}; | ||||
|                 QString second{split_line.last().simplified()}; | ||||
|  | ||||
|                 auto new_test_name = test_name_template.arg(QString::number(test_number), "lessThan").toLatin1().data(); | ||||
|                 QTest::newRow(new_test_name) << first << second << true << false; | ||||
|                 auto new_test_name = test_name_template.arg(QString::number(test_number), "lessThan"); | ||||
|                 m_flex_test_names.append(new_test_name); | ||||
|                 QTest::newRow(m_flex_test_names.last().toLatin1().data()) << first << second << true << false; | ||||
|  | ||||
|                 continue; | ||||
|             } | ||||
| @@ -112,8 +115,9 @@ class VersionTest : public QObject { | ||||
|                 QString first{split_line.first().simplified()}; | ||||
|                 QString second{split_line.last().simplified()}; | ||||
|  | ||||
|                 auto new_test_name = test_name_template.arg(QString::number(test_number), "equals").toLatin1().data(); | ||||
|                 QTest::newRow(new_test_name) << first << second << false << true; | ||||
|                 auto new_test_name = test_name_template.arg(QString::number(test_number), "equals"); | ||||
|                 m_flex_test_names.append(new_test_name); | ||||
|                 QTest::newRow(m_flex_test_names.last().toLatin1().data()) << first << second << false << true; | ||||
|  | ||||
|                 continue; | ||||
|             } | ||||
| @@ -123,8 +127,9 @@ class VersionTest : public QObject { | ||||
|                 QString first{split_line.first().simplified()}; | ||||
|                 QString second{split_line.last().simplified()}; | ||||
|  | ||||
|                 auto new_test_name = test_name_template.arg(QString::number(test_number), "greaterThan").toLatin1().data(); | ||||
|                 QTest::newRow(new_test_name) << first << second << false << false; | ||||
|                 auto new_test_name = test_name_template.arg(QString::number(test_number), "greaterThan"); | ||||
|                 m_flex_test_names.append(new_test_name); | ||||
|                 QTest::newRow(m_flex_test_names.last().toLatin1().data()) << first << second << false << false; | ||||
|  | ||||
|                 continue; | ||||
|             } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Trial97
					Trial97