diff --git a/.clang-tidy b/.clang-tidy
new file mode 100644
index 000000000..11ddc9cfc
--- /dev/null
+++ b/.clang-tidy
@@ -0,0 +1,4 @@
+Checks:
+ - modernize-use-using
+
+SystemHeaders: false
diff --git a/.github/workflows/backport.yml b/.github/workflows/backport.yml
index 08cfb56dd..6f2025551 100644
--- a/.github/workflows/backport.yml
+++ b/.github/workflows/backport.yml
@@ -20,11 +20,11 @@ jobs:
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
+ - uses: actions/checkout@v4
with:
ref: ${{ github.event.pull_request.head.sha }}
- name: Create backport PRs
- uses: korthout/backport-action@v1.4.0
+ uses: korthout/backport-action@v2.1.0
with:
# Config README: https://github.com/korthout/backport-action#backport-action
pull_description: |-
diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml
index ffb3f1214..169e96e6d 100644
--- a/.github/workflows/build.yml
+++ b/.github/workflows/build.yml
@@ -37,56 +37,43 @@ jobs:
fail-fast: false
matrix:
include:
-
- os: ubuntu-20.04
qt_ver: 5
- os: ubuntu-20.04
qt_ver: 6
qt_host: linux
- qt_arch: ''
- qt_version: '6.2.4'
- qt_modules: 'qt5compat qtimageformats'
- qt_tools: ''
+ qt_arch: ""
+ qt_version: "6.2.4"
+ qt_modules: "qt5compat qtimageformats"
+ qt_tools: ""
- os: windows-2022
name: "Windows-MinGW-w64"
msystem: clang64
- vcvars_arch: 'amd64_x86'
-
- - os: windows-2022
- name: "Windows-MSVC-Legacy"
- msystem: ''
- architecture: 'win32'
- vcvars_arch: 'amd64_x86'
- qt_ver: 5
- qt_host: windows
- qt_arch: 'win32_msvc2019'
- qt_version: '5.15.2'
- qt_modules: ''
- qt_tools: 'tools_openssl_x86'
+ vcvars_arch: "amd64_x86"
- os: windows-2022
name: "Windows-MSVC"
- msystem: ''
- architecture: 'x64'
- vcvars_arch: 'amd64'
+ msystem: ""
+ architecture: "x64"
+ vcvars_arch: "amd64"
qt_ver: 6
qt_host: windows
qt_arch: ''
- qt_version: '6.5.2'
+ qt_version: '6.6.0'
qt_modules: 'qt5compat qtimageformats'
qt_tools: ''
- os: windows-2022
name: "Windows-MSVC-arm64"
- msystem: ''
- architecture: 'arm64'
- vcvars_arch: 'amd64_arm64'
+ msystem: ""
+ architecture: "arm64"
+ vcvars_arch: "amd64_arm64"
qt_ver: 6
qt_host: windows
qt_arch: 'win64_msvc2019_arm64'
- qt_version: '6.5.2'
+ qt_version: '6.6.0'
qt_modules: 'qt5compat qtimageformats'
qt_tools: ''
@@ -96,7 +83,7 @@ jobs:
qt_ver: 6
qt_host: mac
qt_arch: ''
- qt_version: '6.5.2'
+ qt_version: '6.6.0'
qt_modules: 'qt5compat qtimageformats'
qt_tools: ''
@@ -105,9 +92,9 @@ jobs:
macosx_deployment_target: 10.13
qt_ver: 5
qt_host: mac
- qt_version: '5.15.2'
- qt_modules: ''
- qt_tools: ''
+ qt_version: "5.15.2"
+ qt_modules: ""
+ qt_tools: ""
runs-on: ${{ matrix.os }}
@@ -125,11 +112,11 @@ jobs:
# PREPARE
##
- name: Checkout
- uses: actions/checkout@v3
+ uses: actions/checkout@v4
with:
- submodules: 'true'
+ submodules: "true"
- - name: 'Setup MSYS2'
+ - name: "Setup MSYS2"
if: runner.os == 'Windows' && matrix.msystem != ''
uses: msys2/setup-msys2@v2
with:
@@ -164,12 +151,12 @@ jobs:
- name: Retrieve ccache cache (Windows MinGW-w64)
if: runner.os == 'Windows' && matrix.msystem != '' && inputs.build_type == 'Debug'
- uses: actions/cache@v3.3.1
+ uses: actions/cache@v3.3.2
with:
path: '${{ github.workspace }}\.ccache'
key: ${{ matrix.os }}-mingw-w64-ccache-${{ github.run_id }}
restore-keys: |
- ${{ matrix.os }}-mingw-w64-ccache
+ ${{ matrix.os }}-mingw-w64-ccache
- name: Setup ccache (Windows MinGW-w64)
if: runner.os == 'Windows' && matrix.msystem != '' && inputs.build_type == 'Debug'
@@ -214,35 +201,35 @@ jobs:
if: runner.os == 'Windows' && matrix.architecture == 'arm64'
uses: jurplel/install-qt-action@v3
with:
- aqtversion: '==3.1.*'
- py7zrversion: '>=0.20.2'
- version: ${{ matrix.qt_version }}
- host: 'windows'
- target: 'desktop'
- arch: ''
- modules: ${{ matrix.qt_modules }}
- tools: ${{ matrix.qt_tools }}
- cache: ${{ inputs.is_qt_cached }}
- cache-key-prefix: host-qt-arm64-windows
- dir: ${{ github.workspace }}\HostQt
- set-env: false
+ aqtversion: "==3.1.*"
+ py7zrversion: ">=0.20.2"
+ version: ${{ matrix.qt_version }}
+ host: "windows"
+ target: "desktop"
+ arch: ""
+ modules: ${{ matrix.qt_modules }}
+ tools: ${{ matrix.qt_tools }}
+ cache: ${{ inputs.is_qt_cached }}
+ cache-key-prefix: host-qt-arm64-windows
+ dir: ${{ github.workspace }}\HostQt
+ set-env: false
- name: Install Qt (macOS, Linux, Qt 6 & Windows MSVC)
if: runner.os == 'Linux' && matrix.qt_ver == 6 || runner.os == 'macOS' || (runner.os == 'Windows' && matrix.msystem == '')
uses: jurplel/install-qt-action@v3
with:
- aqtversion: '==3.1.*'
- py7zrversion: '>=0.20.2'
- version: ${{ matrix.qt_version }}
- host: ${{ matrix.qt_host }}
- target: 'desktop'
- arch: ${{ matrix.qt_arch }}
- modules: ${{ matrix.qt_modules }}
- tools: ${{ matrix.qt_tools }}
- cache: ${{ inputs.is_qt_cached }}
+ aqtversion: "==3.1.*"
+ py7zrversion: ">=0.20.2"
+ version: ${{ matrix.qt_version }}
+ host: ${{ matrix.qt_host }}
+ target: "desktop"
+ arch: ${{ matrix.qt_arch }}
+ modules: ${{ matrix.qt_modules }}
+ tools: ${{ matrix.qt_tools }}
+ cache: ${{ inputs.is_qt_cached }}
- name: Install MSVC (Windows MSVC)
- if: runner.os == 'Windows' # We want this for MinGW builds as well, as we need SignTool
+ if: runner.os == 'Windows' # We want this for MinGW builds as well, as we need SignTool
uses: ilammy/msvc-dev-cmd@v1
with:
vsversion: 2022
@@ -283,12 +270,12 @@ jobs:
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=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
+ 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 -DLauncher_BUILD_ARTIFACT=${{ matrix.name }}-Qt${{ matrix.qt_ver }} -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=official -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 -DLauncher_BUILD_ARTIFACT=${{ matrix.name }}-Qt${{ matrix.qt_ver }}
# https://github.com/ccache/ccache/wiki/MS-Visual-Studio (I coudn't figure out the compiler prefix)
if ("${{ env.CCACHE_VAR }}")
{
@@ -303,7 +290,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=official -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 }} -DLauncher_BUILD_ARTIFACT=Linux-Qt${{ matrix.qt_ver }} -G Ninja
##
# BUILD
@@ -343,7 +330,7 @@ jobs:
- name: Test (Windows MSVC)
if: runner.os == 'Windows' && matrix.msystem == '' && matrix.architecture != 'arm64'
run: |
- ctest -E "^example64|example$" --test-dir build --output-on-failure -C ${{ inputs.build_type }}
+ ctest -E "^example64|example$" --test-dir build --output-on-failure -C ${{ inputs.build_type }}
##
# PACKAGE BUILDS
@@ -384,7 +371,7 @@ jobs:
run: |
cmake --install ${{ env.BUILD_DIR }}
touch ${{ env.INSTALL_DIR }}/manifest.txt
- for l in $(find ${{ env.INSTALL_DIR }} -type f); do l=$(cygpath -u $l); l=${l#$(pwd)/}; l=${l#${{ env.INSTALL_DIR }}/}; l=${l#./}; echo $l; done >> ${{ env.INSTALL_DIR }}/manifest.txt
+ for l in $(find ${{ env.INSTALL_DIR }} -type f); do l=$(cygpath -u $l); l=${l#$(pwd)/}; l=${l#${{ env.INSTALL_DIR }}/}; l=${l#./}; echo $l; done >> ${{ env.INSTALL_DIR }}/manifest.txt
- name: Package (Windows MSVC)
if: runner.os == 'Windows' && matrix.msystem == ''
@@ -401,10 +388,9 @@ jobs:
Get-ChildItem ${{ env.INSTALL_DIR }} -Recurse | ForEach FullName | Resolve-Path -Relative | %{ $_.TrimStart('.\') } | %{ $_.TrimStart('${{ env.INSTALL_DIR }}') } | %{ $_.TrimStart('\') } | Out-File -FilePath ${{ env.INSTALL_DIR }}/manifest.txt
-
- name: Fetch codesign certificate (Windows)
if: runner.os == 'Windows'
- shell: bash # yes, we are not using MSYS2 or PowerShell here
+ shell: bash # yes, we are not using MSYS2 or PowerShell here
run: |
echo '${{ secrets.WINDOWS_CODESIGN_CERT }}' | base64 --decode > codesign.pfx
@@ -414,7 +400,7 @@ jobs:
if (Get-Content ./codesign.pfx){
cd ${{ env.INSTALL_DIR }}
# We ship the exact same executable for portable and non-portable editions, so signing just once is fine
- SignTool sign /fd sha256 /td sha256 /f ../codesign.pfx /p '${{ secrets.WINDOWS_CODESIGN_PASSWORD }}' /tr http://timestamp.digicert.com pollymc.exe pollymc_filelink.exe
+ SignTool sign /fd sha256 /td sha256 /f ../codesign.pfx /p '${{ secrets.WINDOWS_CODESIGN_PASSWORD }}' /tr http://timestamp.digicert.com pollymc.exe pollymc_updater.exe pollymc_filelink.exe
} else {
":warning: Skipped code signing for Windows, as certificate was not present." >> $env:GITHUB_STEP_SUMMARY
}
@@ -506,15 +492,7 @@ jobs:
export LD_LIBRARY_PATH
chmod +x AppImageUpdate-x86_64.AppImage
- ./AppImageUpdate-x86_64.AppImage --appimage-extract
-
- mkdir -p ${{ env.INSTALL_APPIMAGE_DIR }}/usr/optional
- mkdir -p ${{ env.INSTALL_APPIMAGE_DIR }}/usr/plugins
-
- cp -r squashfs-root/usr/bin/* ${{ env.INSTALL_APPIMAGE_DIR }}/usr/bin
- cp -r squashfs-root/usr/lib/* ${{ env.INSTALL_APPIMAGE_DIR }}/usr/lib
- cp -r squashfs-root/usr/optional/* ${{ env.INSTALL_APPIMAGE_DIR }}/usr/optional
- cp -r squashfs-root/usr/optional/* ${{ env.INSTALL_APPIMAGE_DIR }}/usr/plugins
+ cp AppImageUpdate-x86_64.AppImage ${{ env.INSTALL_APPIMAGE_DIR }}/usr/bin
export UPDATE_INFORMATION="gh-releases-zsync|${{ github.repository_owner }}|${{ github.event.repository.name }}|latest|PollyMC-Linux-x86_64.AppImage.zsync"
@@ -522,7 +500,7 @@ jobs:
export SIGN=1
export SIGN_KEY=${{ secrets.GPG_PRIVATE_KEY_ID }}
mkdir -p ~/.gnupg/
- printf "$GPG_PRIVATE_KEY" | base64 --decode > ~/.gnupg/private.key
+ echo "$GPG_PRIVATE_KEY" > ~/.gnupg/private.key
gpg --import ~/.gnupg/private.key
else
echo ":warning: Skipped code signing for Linux AppImage, as gpg key was not present." >> $GITHUB_STEP_SUMMARY
@@ -568,14 +546,14 @@ jobs:
if: runner.os == 'Linux' && matrix.qt_ver != 6
uses: actions/upload-artifact@v3
with:
- name: PollyMC-${{ runner.os }}-${{ env.VERSION }}-${{ inputs.build_type }}
+ name: PollyMC-${{ runner.os }}-Qt5-${{ env.VERSION }}-${{ inputs.build_type }}
path: PollyMC.tar.gz
- name: Upload binary tarball (Linux, portable, Qt 5)
if: runner.os == 'Linux' && matrix.qt_ver != 6
uses: actions/upload-artifact@v3
with:
- name: PollyMC-${{ runner.os }}-Portable-${{ env.VERSION }}-${{ inputs.build_type }}
+ name: PollyMC-${{ runner.os }}-Qt5-Portable-${{ env.VERSION }}-${{ inputs.build_type }}
path: PollyMC-portable.tar.gz
- name: Upload binary tarball (Linux, Qt 6)
@@ -619,10 +597,10 @@ jobs:
options: --privileged
steps:
- name: Checkout
- uses: actions/checkout@v3
+ uses: actions/checkout@v4
if: inputs.build_type == 'Release'
with:
- submodules: 'true'
+ submodules: "true"
- name: Build Flatpak (Linux)
if: inputs.build_type == 'Release'
uses: flatpak/flatpak-github-actions/flatpak-builder@v6
diff --git a/.github/workflows/codeql.yml b/.github/workflows/codeql.yml
index 0cd1f6e40..a77b4ae1e 100644
--- a/.github/workflows/codeql.yml
+++ b/.github/workflows/codeql.yml
@@ -8,7 +8,7 @@ jobs:
steps:
- name: Checkout repository
- uses: actions/checkout@v3
+ uses: actions/checkout@v4
with:
submodules: 'true'
diff --git a/.github/workflows/trigger_builds.yml b/.github/workflows/trigger_builds.yml
index bfab49232..70fda60ed 100644
--- a/.github/workflows/trigger_builds.yml
+++ b/.github/workflows/trigger_builds.yml
@@ -3,24 +3,25 @@ name: Build Application
on:
push:
branches-ignore:
- - 'renovate/**'
+ - "renovate/**"
paths-ignore:
- - '**.md'
- - '**/LICENSE'
- - 'flake.lock'
- - 'packages/**'
- - '.github/ISSUE_TEMPLATE/**'
+ - "**.md"
+ - "**/LICENSE"
+ - "flake.lock"
+ - "packages/**"
+ - ".github/ISSUE_TEMPLATE/**"
+ - ".markdownlint**"
pull_request:
paths-ignore:
- - '**.md'
- - '**/LICENSE'
- - 'flake.lock'
- - 'packages/**'
- - '.github/ISSUE_TEMPLATE/**'
+ - "**.md"
+ - "**/LICENSE"
+ - "flake.lock"
+ - "packages/**"
+ - ".github/ISSUE_TEMPLATE/**"
+ - ".markdownlint**"
workflow_dispatch:
jobs:
-
build_debug:
name: Build Debug
uses: ./.github/workflows/build.yml
@@ -32,3 +33,5 @@ jobs:
WINDOWS_CODESIGN_CERT: ${{ secrets.WINDOWS_CODESIGN_CERT }}
WINDOWS_CODESIGN_PASSWORD: ${{ secrets.WINDOWS_CODESIGN_PASSWORD }}
CACHIX_AUTH_TOKEN: ${{ secrets.CACHIX_AUTH_TOKEN }}
+ GPG_PRIVATE_KEY: ${{ secrets.GPG_PRIVATE_KEY }}
+ GPG_PRIVATE_KEY_ID: ${{ secrets.GPG_PRIVATE_KEY_ID }}
diff --git a/.github/workflows/trigger_release.yml b/.github/workflows/trigger_release.yml
index f19024eb2..0aeadc918 100644
--- a/.github/workflows/trigger_release.yml
+++ b/.github/workflows/trigger_release.yml
@@ -3,10 +3,9 @@ name: Build Application and Make Release
on:
push:
tags:
- - '*'
+ - "*"
jobs:
-
build_release:
name: Build Release
uses: ./.github/workflows/build.yml
@@ -18,6 +17,8 @@ jobs:
WINDOWS_CODESIGN_CERT: ${{ secrets.WINDOWS_CODESIGN_CERT }}
WINDOWS_CODESIGN_PASSWORD: ${{ secrets.WINDOWS_CODESIGN_PASSWORD }}
CACHIX_AUTH_TOKEN: ${{ secrets.CACHIX_AUTH_TOKEN }}
+ GPG_PRIVATE_KEY: ${{ secrets.GPG_PRIVATE_KEY }}
+ GPG_PRIVATE_KEY_ID: ${{ secrets.GPG_PRIVATE_KEY_ID }}
create_release:
needs: build_release
@@ -26,10 +27,10 @@ jobs:
upload_url: ${{ steps.create_release.outputs.upload_url }}
steps:
- name: Checkout
- uses: actions/checkout@v3
+ uses: actions/checkout@v4
with:
- submodules: 'true'
- path: 'PollyMC-source'
+ submodules: "true"
+ path: "PollyMC-source"
- name: Download artifacts
uses: actions/download-artifact@v3
- name: Grab and store version
@@ -41,8 +42,8 @@ jobs:
mv ${{ github.workspace }}/PollyMC-source PollyMC-${{ env.VERSION }}
mv PollyMC-Linux-Qt6-Portable*/PollyMC-portable.tar.gz PollyMC-Linux-Qt6-Portable-${{ env.VERSION }}.tar.gz
mv PollyMC-Linux-Qt6*/PollyMC.tar.gz PollyMC-Linux-Qt6-${{ env.VERSION }}.tar.gz
- mv PollyMC-Linux-Portable*/PollyMC-portable.tar.gz PollyMC-Linux-Portable-${{ env.VERSION }}.tar.gz
- mv PollyMC-Linux*/PollyMC.tar.gz PollyMC-Linux-${{ env.VERSION }}.tar.gz
+ mv PollyMC-Linux-Qt5-Portable*/PollyMC-Qt5-portable.tar.gz PollyMC-Linux-Qt5-Portable-${{ env.VERSION }}.tar.gz
+ mv PollyMC-Linux-Qt5*/PollyMC.tar.gz PollyMC-Linux-Qt5-${{ env.VERSION }}.tar.gz
mv PollyMC-*.AppImage/PollyMC-*.AppImage PollyMC-Linux-x86_64.AppImage
mv PollyMC-*.AppImage.zsync/PollyMC-*.AppImage.zsync PollyMC-Linux-x86_64.AppImage.zsync
mv PollyMC*.flatpak/PollyMC*.flatpak PollyMC-Linux-${{ env.VERSION }}-x86_64.flatpak
@@ -87,8 +88,8 @@ jobs:
draft: true
prerelease: false
files: |
- PollyMC-Linux-${{ env.VERSION }}.tar.gz
- PollyMC-Linux-Portable-${{ env.VERSION }}.tar.gz
+ PollyMC-Linux-Qt5-${{ env.VERSION }}.tar.gz
+ PollyMC-Linux-Qt5-Portable-${{ env.VERSION }}.tar.gz
PollyMC-Linux-x86_64.AppImage
PollyMC-Linux-x86_64.AppImage.zsync
PollyMC-Linux-${{ env.VERSION }}-x86_64.flatpak
@@ -97,9 +98,6 @@ jobs:
PollyMC-Windows-MinGW-w64-${{ env.VERSION }}.zip
PollyMC-Windows-MinGW-w64-Portable-${{ env.VERSION }}.zip
PollyMC-Windows-MinGW-w64-Setup-${{ env.VERSION }}.exe
- PollyMC-Windows-MSVC-Legacy-${{ env.VERSION }}.zip
- PollyMC-Windows-MSVC-Legacy-Portable-${{ env.VERSION }}.zip
- PollyMC-Windows-MSVC-Legacy-Setup-${{ env.VERSION }}.exe
PollyMC-Windows-MSVC-arm64-${{ env.VERSION }}.zip
PollyMC-Windows-MSVC-arm64-Portable-${{ env.VERSION }}.zip
PollyMC-Windows-MSVC-arm64-Setup-${{ env.VERSION }}.exe
@@ -109,4 +107,3 @@ jobs:
PollyMC-macOS-${{ env.VERSION }}.tar.gz
PollyMC-macOS-Legacy-${{ env.VERSION }}.tar.gz
PollyMC-${{ env.VERSION }}.tar.gz
-
diff --git a/.github/workflows/update-flake.yml b/.github/workflows/update-flake.yml
index ce50f4a29..4e8c268f4 100644
--- a/.github/workflows/update-flake.yml
+++ b/.github/workflows/update-flake.yml
@@ -16,8 +16,8 @@ jobs:
runs-on: ubuntu-latest
steps:
- - uses: actions/checkout@v3
- - uses: cachix/install-nix-action@v22
+ - uses: actions/checkout@v4
+ - uses: cachix/install-nix-action@6a9a9e84a173d90b3ffb42c5ddaf9ea033fad011 # v23
- uses: DeterminateSystems/update-flake-lock@v20
with:
diff --git a/CMakeLists.txt b/CMakeLists.txt
index 9e5f4613a..fcbada967 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -188,8 +188,11 @@ set(Launcher_VERSION_NAME4_COMMA "${Launcher_VERSION_MAJOR},${Launcher_VERSION_M
# Build platform.
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.")
+# Github repo URL with releases for updater
+set(Launcher_UPDATER_GITHUB_REPO "https://github.com/fn2006/PollyMC" CACHE STRING "Base github URL for the updater.")
+
+# Name to help updater identify valid artifacts
+set(Launcher_BUILD_ARTIFACT "" CACHE STRING "Artifact name to help the updater identify valid artifacts.")
# The metadata server
set(Launcher_META_URL "https://meta.unmojang.org/v1/" CACHE STRING "URL to fetch Launcher's meta files from.")
@@ -245,6 +248,11 @@ set(Launcher_MSA_CLIENT_ID "" CACHE STRING "Client ID you can get from Microsoft
set(Launcher_CURSEFORGE_API_KEY "" CACHE STRING "API key for the CurseForge platform")
set(Launcher_CURSEFORGE_API_KEY_API_URL "https://cf.polymc.org/api" CACHE STRING "URL to fetch the Curseforge API key from.")
+set(Launcher_COMPILER_NAME ${CMAKE_CXX_COMPILER_ID})
+set(Launcher_COMPILER_VERSION ${CMAKE_CXX_COMPILER_VERSION})
+set(Launcher_COMPILER_TARGET_SYSTEM ${CMAKE_SYSTEM_NAME})
+set(Launcher_COMPILER_TARGET_SYSTEM_VERSION ${CMAKE_SYSTEM_VERSION})
+set(Launcher_COMPILER_TARGET_PROCESSOR ${CMAKE_SYSTEM_PROCESSOR})
#### Check the current Git commit and branch
include(GetGitRevisionDescription)
@@ -339,6 +347,11 @@ add_subdirectory(program_info)
####################################### Install layout #######################################
set(Launcher_ENABLE_UPDATER NO)
+set(Launcher_BUILD_UPDATER NO)
+
+if (NOT APPLE AND (NOT Launcher_UPDATER_GITHUB_REPO STREQUAL "" AND NOT Launcher_BUILD_ARTIFACT STREQUAL ""))
+ set(Launcher_BUILD_UPDATER YES)
+endif()
if(NOT (UNIX AND APPLE))
# Install "portable.txt" if selected component is "portable"
diff --git a/README.md b/README.md
index 145a94684..bef1550e1 100644
--- a/README.md
+++ b/README.md
@@ -49,4 +49,3 @@ There are builds for MacOS in the [releases section](https://github.com/fn2006/P
To build the launcher yourself, follow [the instructions on the Prism Launcher website](https://prismlauncher.org/wiki/development/build-instructions) but clone this repo instead.
-
diff --git a/buildconfig/BuildConfig.cpp.in b/buildconfig/BuildConfig.cpp.in
index cabd84c2c..5013d3b2c 100644
--- a/buildconfig/BuildConfig.cpp.in
+++ b/buildconfig/BuildConfig.cpp.in
@@ -34,6 +34,7 @@
* limitations under the License.
*/
+#include
#include "BuildConfig.h"
#include
@@ -60,8 +61,16 @@ Config::Config()
VERSION_MINOR = @Launcher_VERSION_MINOR@;
BUILD_PLATFORM = "@Launcher_BUILD_PLATFORM@";
+ BUILD_ARTIFACT = "@Launcher_BUILD_ARTIFACT@";
BUILD_DATE = "@Launcher_BUILD_TIMESTAMP@";
- UPDATER_BASE = "@Launcher_UPDATER_BASE@";
+ UPDATER_GITHUB_REPO = "@Launcher_UPDATER_GITHUB_REPO@";
+
+ COMPILER_NAME = "@Launcher_COMPILER_NAME@";
+ COMPILER_VERSION = "@Launcher_COMPILER_VERSION@";
+
+ COMPILER_TARGET_SYSTEM = "@Launcher_COMPILER_TARGET_SYSTEM@";
+ COMPILER_TARGET_SYSTEM_VERSION = "@Launcher_COMPILER_TARGET_SYSTEM_VERSION@";
+ COMPILER_TARGET_SYSTEM_PROCESSOR = "@Launcher_COMPILER_TARGET_PROCESSOR@";
MAC_SPARKLE_PUB_KEY = "@MACOSX_SPARKLE_UPDATE_PUBLIC_KEY@";
MAC_SPARKLE_APPCAST_URL = "@MACOSX_SPARKLE_UPDATE_FEED_URL@";
@@ -69,6 +78,8 @@ Config::Config()
if (!MAC_SPARKLE_PUB_KEY.isEmpty() && !MAC_SPARKLE_APPCAST_URL.isEmpty())
{
UPDATER_ENABLED = true;
+ } else if(!UPDATER_GITHUB_REPO.isEmpty() && !BUILD_ARTIFACT.isEmpty()) {
+ UPDATER_ENABLED = true;
}
GIT_COMMIT = "@Launcher_GIT_COMMIT@";
@@ -89,10 +100,7 @@ Config::Config()
if (GIT_REFSPEC.startsWith("refs/heads/"))
{
VERSION_CHANNEL = GIT_REFSPEC;
- VERSION_CHANNEL.remove("refs/heads/");
- if(!UPDATER_BASE.isEmpty() && !BUILD_PLATFORM.isEmpty()) {
- UPDATER_ENABLED = true;
- }
+ VERSION_CHANNEL.remove("refs/heads/");
}
else if (!GIT_COMMIT.isEmpty())
{
@@ -138,3 +146,16 @@ QString Config::printableVersionString() const
}
return vstr;
}
+
+QString Config::compilerID() const
+{
+ if (COMPILER_VERSION.isEmpty())
+ return COMPILER_NAME;
+ return QStringLiteral("%1 - %2").arg(COMPILER_NAME).arg(COMPILER_VERSION);
+}
+
+QString Config::systemID() const
+{
+ return QStringLiteral("%1 %2 %3").arg(COMPILER_TARGET_SYSTEM, COMPILER_TARGET_SYSTEM_VERSION, COMPILER_TARGET_SYSTEM_PROCESSOR);
+}
+
diff --git a/buildconfig/BuildConfig.h b/buildconfig/BuildConfig.h
index de94f0fca..278da0dfd 100644
--- a/buildconfig/BuildConfig.h
+++ b/buildconfig/BuildConfig.h
@@ -72,11 +72,29 @@ class Config {
/// A short string identifying this build's platform or distribution.
QString BUILD_PLATFORM;
+ /// A short string identifying this build's valid artifacts int he updater. For example, "lin64" or "win32".
+ QString BUILD_ARTIFACT;
+
/// A string containing the build timestamp
QString BUILD_DATE;
+ /// A string identifying the compiler use to build
+ QString COMPILER_NAME;
+
+ /// A string identifying the compiler version used to build
+ QString COMPILER_VERSION;
+
+ /// A string identifying the compiler target system os
+ QString COMPILER_TARGET_SYSTEM;
+
+ /// A String identifying the compiler target system version
+ QString COMPILER_TARGET_SYSTEM_VERSION;
+
+ /// A String identifying the compiler target processor
+ QString COMPILER_TARGET_SYSTEM_PROCESSOR;
+
/// URL for the updater's channel
- QString UPDATER_BASE;
+ QString UPDATER_GITHUB_REPO;
/// The public key used to sign releases for the Sparkle updater appcast
QString MAC_SPARKLE_PUB_KEY;
@@ -187,6 +205,18 @@ class Config {
* \return The version number in string format (major.minor.revision.build).
*/
QString printableVersionString() const;
+
+ /**
+ * \brief Compiler ID String
+ * \return a string of the form "Name - Version" of just "Name" if the version is empty
+ */
+ QString compilerID() const;
+
+ /**
+ * \brief System ID String
+ * \return a string of the form "OS Verison Processor"
+ */
+ QString systemID() const;
};
extern const Config BuildConfig;
diff --git a/cmake/CompilerWarnings.cmake b/cmake/CompilerWarnings.cmake
index 635e54289..c9962c443 100644
--- a/cmake/CompilerWarnings.cmake
+++ b/cmake/CompilerWarnings.cmake
@@ -75,7 +75,6 @@ function(
set(CLANG_WARNINGS
-Wall
-Wextra # reasonable and standard
- -Wextra-semi # Warn about semicolon after in-class function definition.
-Wshadow # warn the user if a variable declaration shadows one from a parent context
-Wnon-virtual-dtor # warn the user if a class with virtual functions has a non-virtual destructor. This helps
# catch hard to track down memory errors
@@ -90,6 +89,10 @@ function(
-Wdouble-promotion # warn if float is implicit promoted to double
-Wformat=2 # warn on security issues around functions that format output (ie printf)
-Wimplicit-fallthrough # warn on statements that fallthrough without an explicit annotation
+ # -Wgnu-zero-variadic-macro-arguments (part of -pedantic) is triggered by every qCDebug() call and therefore results
+ # in a lot of noise. This warning is only notifying us that clang is emulating the GCC behaviour
+ # instead of the exact standard wording so we can safely ignore it
+ -Wno-gnu-zero-variadic-macro-arguments
)
endif()
diff --git a/flake.lock b/flake.lock
index b1486ea69..a520c27e5 100644
--- a/flake.lock
+++ b/flake.lock
@@ -3,11 +3,11 @@
"flake-compat": {
"flake": false,
"locked": {
- "lastModified": 1673956053,
- "narHash": "sha256-4gtG9iQuiKITOjNQQeQIpoIB6b16fm+504Ch3sNKLd8=",
+ "lastModified": 1696426674,
+ "narHash": "sha256-kvjfFW7WAETZlt09AgDn1MrtKzP7t90Vf7vypd3OL1U=",
"owner": "edolstra",
"repo": "flake-compat",
- "rev": "35bb57c0c8d8b62bbfd284272c928ceb64ddbde9",
+ "rev": "0f9255e01c2351cc7d116c072cb317785dd33b33",
"type": "github"
},
"original": {
@@ -21,11 +21,11 @@
"nixpkgs-lib": "nixpkgs-lib"
},
"locked": {
- "lastModified": 1693611461,
- "narHash": "sha256-aPODl8vAgGQ0ZYFIRisxYG5MOGSkIczvu2Cd8Gb9+1Y=",
+ "lastModified": 1698882062,
+ "narHash": "sha256-HkhafUayIqxXyHH1X8d9RDl1M2CkFgZLjKD3MzabiEo=",
"owner": "hercules-ci",
"repo": "flake-parts",
- "rev": "7f53fdb7bdc5bb237da7fefef12d099e4fd611ca",
+ "rev": "8c9fa2545007b49a5db5f650ae91f227672c3877",
"type": "github"
},
"original": {
@@ -89,13 +89,28 @@
"type": "github"
}
},
+ "nix-filter": {
+ "locked": {
+ "lastModified": 1694857738,
+ "narHash": "sha256-bxxNyLHjhu0N8T3REINXQ2ZkJco0ABFPn6PIe2QUfqo=",
+ "owner": "numtide",
+ "repo": "nix-filter",
+ "rev": "41fd48e00c22b4ced525af521ead8792402de0ea",
+ "type": "github"
+ },
+ "original": {
+ "owner": "numtide",
+ "repo": "nix-filter",
+ "type": "github"
+ }
+ },
"nixpkgs": {
"locked": {
- "lastModified": 1693626178,
- "narHash": "sha256-Rpiy6lIOu4zny8tfGuIeN1ji9eSz9nPmm9yBhh/4IOM=",
+ "lastModified": 1699094435,
+ "narHash": "sha256-YLZ5/KKZ1PyLrm2MO8UxRe4H3M0/oaYqNhSlq6FDeeA=",
"owner": "nixos",
"repo": "nixpkgs",
- "rev": "bfb7dfec93f3b5d7274db109f2990bc889861caf",
+ "rev": "9d5d25bbfe8c0297ebe85324addcb5020ed1a454",
"type": "github"
},
"original": {
@@ -108,11 +123,11 @@
"nixpkgs-lib": {
"locked": {
"dir": "lib",
- "lastModified": 1693471703,
- "narHash": "sha256-0l03ZBL8P1P6z8MaSDS/MvuU8E75rVxe5eE1N6gxeTo=",
+ "lastModified": 1698611440,
+ "narHash": "sha256-jPjHjrerhYDy3q9+s5EAsuhyhuknNfowY6yt6pjn9pc=",
"owner": "NixOS",
"repo": "nixpkgs",
- "rev": "3e52e76b70d5508f3cec70b882a29199f4d1ee85",
+ "rev": "0cbe9f69c234a7700596e943bfae7ef27a31b735",
"type": "github"
},
"original": {
@@ -138,11 +153,11 @@
]
},
"locked": {
- "lastModified": 1692274144,
- "narHash": "sha256-BxTQuRUANQ81u8DJznQyPmRsg63t4Yc+0kcyq6OLz8s=",
+ "lastModified": 1698852633,
+ "narHash": "sha256-Hsc/cCHud8ZXLvmm8pxrXpuaPEeNaaUttaCvtdX/Wug=",
"owner": "cachix",
"repo": "pre-commit-hooks.nix",
- "rev": "7e3517c03d46159fdbf8c0e5c97f82d5d4b0c8fa",
+ "rev": "dec10399e5b56aa95fcd530e0338be72ad6462a0",
"type": "github"
},
"original": {
@@ -156,6 +171,7 @@
"flake-compat": "flake-compat",
"flake-parts": "flake-parts",
"libnbtplusplus": "libnbtplusplus",
+ "nix-filter": "nix-filter",
"nixpkgs": "nixpkgs",
"pre-commit-hooks": "pre-commit-hooks"
}
diff --git a/flake.nix b/flake.nix
index d45282aa6..afb0ec63a 100644
--- a/flake.nix
+++ b/flake.nix
@@ -4,6 +4,7 @@
inputs = {
nixpkgs.url = "github:nixos/nixpkgs/nixpkgs-unstable";
flake-parts.url = "github:hercules-ci/flake-parts";
+ nix-filter.url = "github:numtide/nix-filter";
pre-commit-hooks = {
url = "github:cachix/pre-commit-hooks.nix";
inputs.nixpkgs.follows = "nixpkgs";
@@ -20,12 +21,14 @@
};
};
- outputs = inputs:
- inputs.flake-parts.lib.mkFlake
- {inherit inputs;}
- {
+ outputs = {
+ flake-parts,
+ pre-commit-hooks,
+ ...
+ } @ inputs:
+ flake-parts.lib.mkFlake {inherit inputs;} {
imports = [
- inputs.pre-commit-hooks.flakeModule
+ pre-commit-hooks.flakeModule
./nix/dev.nix
./nix/distribution.nix
diff --git a/launcher/Application.cpp b/launcher/Application.cpp
index 49e99d10b..3ca8ed66c 100644
--- a/launcher/Application.cpp
+++ b/launcher/Application.cpp
@@ -58,6 +58,7 @@
#include "ui/pages/global/APIPage.h"
#include "ui/pages/global/AccountListPage.h"
#include "ui/pages/global/CustomCommandsPage.h"
+#include "ui/pages/global/EnvironmentVariablesPage.h"
#include "ui/pages/global/ExternalToolsPage.h"
#include "ui/pages/global/JavaPage.h"
#include "ui/pages/global/LanguagePage.h"
@@ -125,6 +126,7 @@
#include
#include
+#include
#include
#ifdef Q_OS_LINUX
@@ -133,9 +135,13 @@
#include "gamemode_client.h"
#endif
-#if defined(Q_OS_MAC) && defined(SPARKLE_ENABLED)
+#if defined(Q_OS_MAC)
+#if defined(SPARKLE_ENABLED)
#include "updater/MacSparkleUpdater.h"
#endif
+#else
+#include "updater/PrismExternalUpdater.h"
+#endif
#if defined Q_OS_WIN32
#include "WindowsConsole.h"
@@ -167,6 +173,34 @@ void appDebugOutput(QtMsgType type, const QMessageLogContext& context, const QSt
} // namespace
+std::tuple read_lock_File(const QString& path)
+{
+ auto contents = QString(FS::read(path));
+ auto lines = contents.split('\n');
+
+ QDateTime timestamp;
+ QString from, to, target, data_path;
+ for (auto line : lines) {
+ auto index = line.indexOf("=");
+ if (index < 0)
+ continue;
+ auto left = line.left(index);
+ auto right = line.mid(index + 1);
+ if (left.toLower() == "timestamp") {
+ timestamp = QDateTime::fromString(right, Qt::ISODate);
+ } else if (left.toLower() == "from") {
+ from = right;
+ } else if (left.toLower() == "to") {
+ to = right;
+ } else if (left.toLower() == "target") {
+ target = right;
+ } else if (left.toLower() == "data_path") {
+ data_path = right;
+ }
+ }
+ return std::make_tuple(timestamp, from, to, target, data_path);
+}
+
Application::Application(int& argc, char** argv) : QApplication(argc, argv)
{
#if defined Q_OS_WIN32
@@ -299,6 +333,7 @@ Application::Application(int& argc, char** argv) : QApplication(argc, argv)
.arg(dataPath));
return;
}
+ m_dataPath = dataPath;
/*
* Establish the mechanism for communication with an already running PrismLauncher that uses the same data path.
@@ -453,11 +488,16 @@ Application::Application(int& argc, char** argv) : QApplication(argc, argv)
}
{
- qDebug() << BuildConfig.LAUNCHER_DISPLAYNAME << ", (c) 2013-2021 " << BuildConfig.LAUNCHER_COPYRIGHT;
+ qDebug() << qPrintable(BuildConfig.LAUNCHER_DISPLAYNAME) << ", (c) 2022-2023 "
+ << qPrintable(QString(BuildConfig.LAUNCHER_COPYRIGHT).replace("\n", ", "));
qDebug() << "Version : " << BuildConfig.printableVersionString();
qDebug() << "Platform : " << BuildConfig.BUILD_PLATFORM;
qDebug() << "Git commit : " << BuildConfig.GIT_COMMIT;
qDebug() << "Git refspec : " << BuildConfig.GIT_REFSPEC;
+ qDebug() << "Compiled for : " << BuildConfig.systemID();
+ qDebug() << "Compiled by : " << BuildConfig.compilerID();
+ qDebug() << "Build Artifact : " << BuildConfig.BUILD_ARTIFACT;
+ qDebug() << "Updates Enabled : " << (updaterEnabled() ? "Yes" : "No");
if (adjustedBy.size()) {
qDebug() << "Work dir before adjustment : " << origcwdPath;
qDebug() << "Work dir after adjustment : " << QDir::currentPath();
@@ -506,6 +546,9 @@ Application::Application(int& argc, char** argv) : QApplication(argc, argv)
m_settings->registerSetting("MenuBarInsteadOfToolBar", false);
+ m_settings->registerSetting("NumberOfConcurrentTasks", 10);
+ m_settings->registerSetting("NumberOfConcurrentDownloads", 6);
+
QString defaultMonospace;
int defaultSize = 11;
#ifdef Q_OS_WIN32
@@ -582,6 +625,9 @@ Application::Application(int& argc, char** argv) : QApplication(argc, argv)
m_settings->registerSetting("IgnoreJavaCompatibility", false);
m_settings->registerSetting("IgnoreJavaWizard", false);
+ // Legacy settings
+ m_settings->registerSetting("OnlineFixes", false);
+
// Native library workarounds
m_settings->registerSetting("UseNativeOpenAL", false);
m_settings->registerSetting("CustomOpenALPath", "");
@@ -601,6 +647,7 @@ Application::Application(int& argc, char** argv) : QApplication(argc, argv)
// Minecraft mods
m_settings->registerSetting("ModMetadataDisabled", false);
+ m_settings->registerSetting("ModDependenciesDisabled", false);
// Missing authlib-injector behavior
m_settings->registerSetting("MissingAuthlibInjectorBehavior", MissingAuthlibInjectorBehavior::Ask);
@@ -680,6 +727,8 @@ Application::Application(int& argc, char** argv) : QApplication(argc, argv)
m_settings->registerSetting("CloseAfterLaunch", false);
m_settings->registerSetting("QuitAfterGameStop", false);
+ m_settings->registerSetting("Env", QVariant(QMap()));
+
// Custom Microsoft Authentication Client ID
m_settings->registerSetting("MSAClientIDOverride", "");
@@ -706,6 +755,7 @@ Application::Application(int& argc, char** argv) : QApplication(argc, argv)
m_globalSettingsProvider->addPage();
m_globalSettingsProvider->addPage();
m_globalSettingsProvider->addPage();
+ m_globalSettingsProvider->addPage();
m_globalSettingsProvider->addPage();
m_globalSettingsProvider->addPage();
m_globalSettingsProvider->addPage();
@@ -742,15 +792,6 @@ Application::Application(int& argc, char** argv) : QApplication(argc, argv)
qDebug() << "<> Translations loaded.";
}
- // initialize the updater
- if (BuildConfig.UPDATER_ENABLED) {
- qDebug() << "Initializing updater";
-#if defined(Q_OS_MAC) && defined(SPARKLE_ENABLED)
- m_updater.reset(new MacSparkleUpdater());
-#endif
- qDebug() << "<> Updater started.";
- }
-
// Instance icons
{
auto setting = APPLICATION->settings()->getSetting("IconsDir");
@@ -853,6 +894,107 @@ Application::Application(int& argc, char** argv) : QApplication(argc, argv)
detectLibraries();
+ // check update locks
+ {
+ auto update_log_path = FS::PathCombine(m_dataPath, "logs", "prism_launcher_update.log");
+
+ auto update_lock = QFileInfo(FS::PathCombine(m_dataPath, ".prism_launcher_update.lock"));
+ if (update_lock.exists()) {
+ auto [timestamp, from, to, target, data_path] = read_lock_File(update_lock.absoluteFilePath());
+ auto infoMsg = tr("This installation has a update lock file present at: %1\n"
+ "\n"
+ "Timestamp: %2\n"
+ "Updating from version %3 to %4\n"
+ "Target install path: %5\n"
+ "Data Path: %6"
+ "\n"
+ "This likely means that a update attempt failed. Please ensure your installation is in working order before "
+ "proceeding.\n"
+ "Check the Prism Launcher updater log at: \n"
+ "%7\n"
+ "for details on the last update attempt.\n"
+ "\n"
+ "To delete this lock and proceed select \"Ignore\" below.")
+ .arg(update_lock.absoluteFilePath())
+ .arg(timestamp.toString(Qt::ISODate), from, to, target, data_path)
+ .arg(update_log_path);
+ auto msgBox = QMessageBox(QMessageBox::Warning, tr("Update In Progress"), infoMsg, QMessageBox::Ignore | QMessageBox::Abort);
+ msgBox.setDefaultButton(QMessageBox::Abort);
+ msgBox.setModal(true);
+ msgBox.setDetailedText(FS::read(update_log_path));
+ msgBox.setMinimumWidth(460);
+ msgBox.adjustSize();
+ auto res = msgBox.exec();
+ switch (res) {
+ case QMessageBox::Ignore: {
+ FS::deletePath(update_lock.absoluteFilePath());
+ break;
+ }
+ case QMessageBox::Abort:
+ [[fallthrough]];
+ default: {
+ qDebug() << "Exiting because update lockfile is present";
+ QMetaObject::invokeMethod(
+ this, []() { exit(1); }, Qt::QueuedConnection);
+ return;
+ }
+ }
+ }
+
+ auto update_fail_marker = QFileInfo(FS::PathCombine(m_dataPath, ".prism_launcher_update.fail"));
+ if (update_fail_marker.exists()) {
+ auto infoMsg = tr("An update attempt failed\n"
+ "\n"
+ "Please ensure your installation is in working order before "
+ "proceeding.\n"
+ "Check the Prism Launcher updater log at: \n"
+ "%1\n"
+ "for details on the last update attempt.")
+ .arg(update_log_path);
+ auto msgBox = QMessageBox(QMessageBox::Warning, tr("Update Failed"), infoMsg, QMessageBox::Ignore | QMessageBox::Abort);
+ msgBox.setDefaultButton(QMessageBox::Abort);
+ msgBox.setModal(true);
+ msgBox.setDetailedText(FS::read(update_log_path));
+ msgBox.setMinimumWidth(460);
+ msgBox.adjustSize();
+ auto res = msgBox.exec();
+ switch (res) {
+ case QMessageBox::Ignore: {
+ FS::deletePath(update_fail_marker.absoluteFilePath());
+ break;
+ }
+ case QMessageBox::Abort:
+ [[fallthrough]];
+ default: {
+ qDebug() << "Exiting because update lockfile is present";
+ QMetaObject::invokeMethod(
+ this, []() { exit(1); }, Qt::QueuedConnection);
+ return;
+ }
+ }
+ }
+
+ auto update_success_marker = QFileInfo(FS::PathCombine(m_dataPath, ".prism_launcher_update.success"));
+ if (update_success_marker.exists()) {
+ auto infoMsg = tr("Update succeeded\n"
+ "\n"
+ "You are now running %1 .\n"
+ "Check the Prism Launcher updater log at: \n"
+ "%1\n"
+ "for details.")
+ .arg(BuildConfig.printableVersionString())
+ .arg(update_log_path);
+ auto msgBox = new QMessageBox(QMessageBox::Information, tr("Update Succeeded"), infoMsg, QMessageBox::Ok);
+ msgBox->setDefaultButton(QMessageBox::Ok);
+ msgBox->setDetailedText(FS::read(update_log_path));
+ msgBox->setAttribute(Qt::WA_DeleteOnClose);
+ msgBox->setMinimumWidth(460);
+ msgBox->adjustSize();
+ msgBox->open();
+ FS::deletePath(update_success_marker.absoluteFilePath());
+ }
+ }
+
if (createSetupWizard()) {
return;
}
@@ -921,6 +1063,26 @@ bool Application::createSetupWizard()
return false;
}
+bool Application::updaterEnabled()
+{
+#if defined(Q_OS_MAC)
+ return BuildConfig.UPDATER_ENABLED;
+#else
+ return BuildConfig.UPDATER_ENABLED && QFileInfo(FS::PathCombine(m_rootPath, updaterBinaryName())).isFile();
+#endif
+}
+
+QString Application::updaterBinaryName()
+{
+ auto exe_name = QStringLiteral("%1_updater").arg(BuildConfig.LAUNCHER_APP_BINARY_NAME);
+#if defined Q_OS_WIN32
+ exe_name.append(".exe");
+#else
+ exe_name.prepend("bin/");
+#endif
+ return exe_name;
+}
+
bool Application::event(QEvent* event)
{
#ifdef Q_OS_MACOS
@@ -1001,6 +1163,20 @@ void Application::performMainStartupAction()
showMainWindow(false);
qDebug() << "<> Main window shown.";
}
+
+ // initialize the updater
+ if (updaterEnabled()) {
+ qDebug() << "Initializing updater";
+#ifdef Q_OS_MAC
+#if defined(SPARKLE_ENABLED)
+ m_updater.reset(new MacSparkleUpdater());
+#endif
+#else
+ m_updater.reset(new PrismExternalUpdater(m_mainWindow, m_rootPath, m_dataPath));
+#endif
+ qDebug() << "<> Updater started.";
+ }
+
if (!m_urlsToImport.isEmpty()) {
qDebug() << "<> Importing from url:" << m_urlsToImport;
m_mainWindow->processURLs(m_urlsToImport);
diff --git a/launcher/Application.h b/launcher/Application.h
index b227bb813..7669e08ec 100644
--- a/launcher/Application.h
+++ b/launcher/Application.h
@@ -159,6 +159,9 @@ class Application : public QApplication {
/// this is the root of the 'installation'. Used for automatic updates
const QString& root() { return m_rootPath; }
+ /// the data path the application is using
+ const QString& dataRoot() { return m_dataPath; }
+
bool isPortable() { return m_portable; }
const Capabilities capabilities() { return m_capabilities; }
@@ -179,6 +182,9 @@ class Application : public QApplication {
int suitableMaxMem();
+ bool updaterEnabled();
+ QString updaterBinaryName();
+
QUrl normalizeImportUrl(QString const& url);
signals:
@@ -244,6 +250,7 @@ class Application : public QApplication {
QMap> m_profilers;
QString m_rootPath;
+ QString m_dataPath;
Status m_status = Application::StartingUp;
Capabilities m_capabilities;
bool m_portable = false;
diff --git a/launcher/BaseInstance.cpp b/launcher/BaseInstance.cpp
index 725036395..33dc3f741 100644
--- a/launcher/BaseInstance.cpp
+++ b/launcher/BaseInstance.cpp
@@ -388,7 +388,7 @@ QString BaseInstance::name() const
QString BaseInstance::windowTitle() const
{
- return BuildConfig.LAUNCHER_DISPLAYNAME + ": " + name().replace(QRegularExpression("\\s+"), " ");
+ return BuildConfig.LAUNCHER_DISPLAYNAME + ": " + name();
}
// FIXME: why is this here? move it to MinecraftInstance!!!
diff --git a/launcher/BaseInstance.h b/launcher/BaseInstance.h
index 47fff5957..f4ed9113c 100644
--- a/launcher/BaseInstance.h
+++ b/launcher/BaseInstance.h
@@ -64,7 +64,7 @@ class LaunchTask;
class BaseInstance;
// pointer for lazy people
-typedef std::shared_ptr InstancePtr;
+using InstancePtr = std::shared_ptr;
/*!
* \brief Base class for instances.
diff --git a/launcher/BaseVersionList.h b/launcher/BaseVersionList.h
index fe1550c21..231887c4e 100644
--- a/launcher/BaseVersionList.h
+++ b/launcher/BaseVersionList.h
@@ -51,7 +51,7 @@ class BaseVersionList : public QAbstractListModel {
ArchitectureRole,
SortRole
};
- typedef QList RoleList;
+ using RoleList = QList;
explicit BaseVersionList(QObject* parent = 0);
diff --git a/launcher/CMakeLists.txt b/launcher/CMakeLists.txt
index f66b81859..f88309347 100644
--- a/launcher/CMakeLists.txt
+++ b/launcher/CMakeLists.txt
@@ -144,6 +144,7 @@ set(NET_SOURCES
net/HeaderProxy.h
net/RawHeaderProxy.h
net/ApiHeaderProxy.h
+ net/StaticHeaderProxy.h
net/ApiDownload.h
net/ApiDownload.cpp
net/ApiUpload.cpp
@@ -186,6 +187,11 @@ set(MAC_UPDATE_SOURCES
updater/MacSparkleUpdater.mm
)
+set(PRISM_UPDATE_SOURCES
+ updater/PrismExternalUpdater.h
+ updater/PrismExternalUpdater.cpp
+)
+
# Backend for the news bar... there's usually no news.
set(NEWS_SOURCES
# News System
@@ -606,6 +612,63 @@ set(LINKEXE_SOURCES
DesktopServices.cpp
)
+set(PRISMUPDATER_SOURCES
+ updater/prismupdater/PrismUpdater.h
+ updater/prismupdater/PrismUpdater.cpp
+ updater/prismupdater/UpdaterDialogs.h
+ updater/prismupdater/UpdaterDialogs.cpp
+ updater/prismupdater/GitHubRelease.h
+ updater/prismupdater/GitHubRelease.cpp
+
+ Json.h
+ Json.cpp
+ FileSystem.h
+ FileSystem.cpp
+ StringUtils.h
+ StringUtils.cpp
+ DesktopServices.h
+ DesktopServices.cpp
+ Version.h
+ Version.cpp
+ Markdown.h
+ Markdown.cpp
+
+ # Zip
+ MMCZip.h
+ MMCZip.cpp
+
+ # Time
+ MMCTime.h
+ MMCTime.cpp
+
+ net/ByteArraySink.h
+ net/ChecksumValidator.h
+ net/Download.cpp
+ net/Download.h
+ net/FileSink.cpp
+ net/FileSink.h
+ net/HttpMetaCache.cpp
+ net/HttpMetaCache.h
+ net/Logging.h
+ net/Logging.cpp
+ net/NetAction.h
+ net/NetRequest.cpp
+ net/NetRequest.h
+ net/NetJob.cpp
+ net/NetJob.h
+ net/NetUtils.h
+ net/Sink.h
+ net/Validator.h
+ net/HeaderProxy.h
+ net/RawHeaderProxy.h
+
+ ui/dialogs/ProgressDialog.cpp
+ ui/dialogs/ProgressDialog.h
+ ui/widgets/SubTaskProgressBar.h
+ ui/widgets/SubTaskProgressBar.cpp
+
+)
+
######## Logging categories ########
ecm_qt_declare_logging_category(CORE_SOURCES
@@ -703,6 +766,8 @@ set(LOGIC_SOURCES
if(APPLE AND Launcher_ENABLE_UPDATER)
set (LOGIC_SOURCES ${LOGIC_SOURCES} ${MAC_UPDATE_SOURCES})
+else()
+ set (LOGIC_SOURCES ${LOGIC_SOURCES} ${PRISM_UPDATE_SOURCES})
endif()
SET(LAUNCHER_SOURCES
@@ -852,6 +917,8 @@ SET(LAUNCHER_SOURCES
ui/pages/global/AccountListPage.h
ui/pages/global/CustomCommandsPage.cpp
ui/pages/global/CustomCommandsPage.h
+ ui/pages/global/EnvironmentVariablesPage.cpp
+ ui/pages/global/EnvironmentVariablesPage.h
ui/pages/global/ExternalToolsPage.cpp
ui/pages/global/ExternalToolsPage.h
ui/pages/global/JavaPage.cpp
@@ -941,6 +1008,9 @@ SET(LAUNCHER_SOURCES
ui/pages/modplatform/ImportPage.cpp
ui/pages/modplatform/ImportPage.h
+ ui/pages/modplatform/OptionalModDialog.cpp
+ ui/pages/modplatform/OptionalModDialog.h
+
ui/pages/modplatform/modrinth/ModrinthResourceModels.cpp
ui/pages/modplatform/modrinth/ModrinthResourceModels.h
ui/pages/modplatform/modrinth/ModrinthResourcePages.cpp
@@ -1011,6 +1081,8 @@ SET(LAUNCHER_SOURCES
ui/widgets/Common.h
ui/widgets/CustomCommands.cpp
ui/widgets/CustomCommands.h
+ ui/widgets/EnvironmentVariables.cpp
+ ui/widgets/EnvironmentVariables.h
ui/widgets/DropLabel.cpp
ui/widgets/DropLabel.h
ui/widgets/FocusLineEdit.cpp
@@ -1069,6 +1141,15 @@ SET(LAUNCHER_SOURCES
ui/instanceview/VisualGroup.h
)
+if (NOT Apple)
+set(LAUNCHER_SOURCES
+ ${LAUNCHER_SOURCES}
+
+ ui/dialogs/UpdateAvailableDialog.h
+ ui/dialogs/UpdateAvailableDialog.cpp
+)
+endif()
+
if(WIN32)
set(LAUNCHER_SOURCES
WindowsConsole.cpp
@@ -1108,10 +1189,12 @@ qt_wrap_ui(LAUNCHER_UI
ui/pages/modplatform/import_ftb/ImportFTBPage.ui
ui/pages/modplatform/ImportPage.ui
ui/pages/modplatform/ftb/FtbPage.ui
+ ui/pages/modplatform/OptionalModDialog.ui
ui/pages/modplatform/modrinth/ModrinthPage.ui
ui/pages/modplatform/technic/TechnicPage.ui
ui/widgets/InstanceCardWidget.ui
ui/widgets/CustomCommands.ui
+ ui/widgets/EnvironmentVariables.ui
ui/widgets/InfoFrame.ui
ui/widgets/ModFilterWidget.ui
ui/widgets/SubTaskProgressBar.ui
@@ -1141,6 +1224,14 @@ qt_wrap_ui(LAUNCHER_UI
ui/dialogs/ChooseProviderDialog.ui
)
+qt_wrap_ui(PRISM_UPDATE_UI
+ ui/dialogs/UpdateAvailableDialog.ui
+)
+
+if (NOT Apple)
+ set (LAUNCHER_UI ${LAUNCHER_UI} ${PRISM_UPDATE_UI})
+endif()
+
qt_add_resources(LAUNCHER_RESOURCES
resources/backgrounds/backgrounds.qrc
resources/multimc/multimc.qrc
@@ -1157,6 +1248,12 @@ qt_add_resources(LAUNCHER_RESOURCES
../${Launcher_Branding_LogoQRC}
)
+qt_wrap_ui(PRISMUPDATER_UI
+ updater/prismupdater/SelectReleaseDialog.ui
+ ui/widgets/SubTaskProgressBar.ui
+ ui/dialogs/ProgressDialog.ui
+)
+
######## Windows resource files ########
if(WIN32)
set(LAUNCHER_RCS ${CMAKE_CURRENT_BINARY_DIR}/../${Launcher_Branding_WindowsRC})
@@ -1166,12 +1263,16 @@ include(CompilerWarnings)
# Add executable
add_library(Launcher_logic STATIC ${LOGIC_SOURCES} ${LAUNCHER_SOURCES} ${LAUNCHER_UI} ${LAUNCHER_RESOURCES})
+if(BUILD_TESTING)
+target_compile_definitions(Launcher_logic PUBLIC LAUNCHER_TEST)
+endif()
set_project_warnings(Launcher_logic
"${Launcher_MSVC_WARNINGS}"
"${Launcher_CLANG_WARNINGS}"
"${Launcher_GCC_WARNINGS}")
target_compile_definitions(Launcher_logic PUBLIC LAUNCHER_APPLICATION)
target_include_directories(Launcher_logic PUBLIC ${CMAKE_CURRENT_SOURCE_DIR})
+target_compile_definitions(Launcher_logic PUBLIC LAUNCHER_APPLICATION)
target_link_libraries(Launcher_logic
systeminfo
Launcher_murmur2
@@ -1253,7 +1354,45 @@ install(TARGETS ${Launcher_Name}
FRAMEWORK DESTINATION ${FRAMEWORK_DEST_DIR} COMPONENT Runtime
)
-if(WIN32)
+if(Launcher_BUILD_UPDATER)
+ # Updater
+ add_library(prism_updater_logic STATIC ${PRISMUPDATER_SOURCES} ${TASKS_SOURCES} ${PRISMUPDATER_UI})
+ target_include_directories(prism_updater_logic PUBLIC ${CMAKE_CURRENT_SOURCE_DIR})
+ target_link_libraries(prism_updater_logic
+ QuaZip::QuaZip
+ ${ZLIB_LIBRARIES}
+ systeminfo
+ BuildConfig
+ ghcFilesystem::ghc_filesystem
+ Qt${QT_VERSION_MAJOR}::Widgets
+ Qt${QT_VERSION_MAJOR}::Core
+ Qt${QT_VERSION_MAJOR}::Network
+ ${Launcher_QT_LIBS}
+ cmark::cmark
+ Katabasis
+ )
+
+ add_executable("${Launcher_Name}_updater" WIN32 updater/prismupdater/updater_main.cpp)
+ target_sources("${Launcher_Name}_updater" PRIVATE updater/prismupdater/updater.exe.manifest)
+ target_link_libraries("${Launcher_Name}_updater" prism_updater_logic)
+
+ if(DEFINED Launcher_APP_BINARY_NAME)
+ set_target_properties("${Launcher_Name}_updater" PROPERTIES OUTPUT_NAME "${Launcher_APP_BINARY_NAME}_updater")
+ endif()
+ if(DEFINED Launcher_BINARY_RPATH)
+ SET_TARGET_PROPERTIES("${Launcher_Name}_updater" PROPERTIES INSTALL_RPATH "${Launcher_BINARY_RPATH}")
+ endif()
+
+ install(TARGETS "${Launcher_Name}_updater"
+ BUNDLE DESTINATION "." COMPONENT Runtime
+ LIBRARY DESTINATION ${LIBRARY_DEST_DIR} COMPONENT Runtime
+ RUNTIME DESTINATION ${BINARY_DEST_DIR} COMPONENT Runtime
+ FRAMEWORK DESTINATION ${FRAMEWORK_DEST_DIR} COMPONENT Runtime
+ )
+endif()
+
+if(WIN32 OR (DEFINED Launcher_BUILD_FILELINKER AND Launcher_BUILD_FILELINKER))
+ # File link
add_library(filelink_logic STATIC ${LINKEXE_SOURCES})
set_project_warnings(filelink_logic
"${Launcher_MSVC_WARNINGS}"
@@ -1272,7 +1411,7 @@ if(WIN32)
${Launcher_QT_LIBS}
)
- add_executable("${Launcher_Name}_filelink" WIN32 filelink/main.cpp)
+ add_executable("${Launcher_Name}_filelink" WIN32 filelink/filelink_main.cpp)
target_sources("${Launcher_Name}_filelink" PRIVATE filelink/filelink.exe.manifest)
diff --git a/launcher/FileSystem.cpp b/launcher/FileSystem.cpp
index defb2cb9e..c7d5f85fa 100644
--- a/launcher/FileSystem.cpp
+++ b/launcher/FileSystem.cpp
@@ -123,26 +123,35 @@ namespace fs = ghc::filesystem;
#if defined(__MINGW32__)
-typedef struct _DUPLICATE_EXTENTS_DATA {
+struct _DUPLICATE_EXTENTS_DATA {
HANDLE FileHandle;
LARGE_INTEGER SourceFileOffset;
LARGE_INTEGER TargetFileOffset;
LARGE_INTEGER ByteCount;
-} DUPLICATE_EXTENTS_DATA, *PDUPLICATE_EXTENTS_DATA;
+};
-typedef struct _FSCTL_GET_INTEGRITY_INFORMATION_BUFFER {
+using DUPLICATE_EXTENTS_DATA = _DUPLICATE_EXTENTS_DATA;
+using PDUPLICATE_EXTENTS_DATA = _DUPLICATE_EXTENTS_DATA*;
+
+struct _FSCTL_GET_INTEGRITY_INFORMATION_BUFFER {
WORD ChecksumAlgorithm; // Checksum algorithm. e.g. CHECKSUM_TYPE_UNCHANGED, CHECKSUM_TYPE_NONE, CHECKSUM_TYPE_CRC32
WORD Reserved; // Must be 0
DWORD Flags; // FSCTL_INTEGRITY_FLAG_xxx
DWORD ChecksumChunkSizeInBytes;
DWORD ClusterSizeInBytes;
-} FSCTL_GET_INTEGRITY_INFORMATION_BUFFER, *PFSCTL_GET_INTEGRITY_INFORMATION_BUFFER;
+};
+
+using FSCTL_GET_INTEGRITY_INFORMATION_BUFFER = _FSCTL_GET_INTEGRITY_INFORMATION_BUFFER;
+using PFSCTL_GET_INTEGRITY_INFORMATION_BUFFER = _FSCTL_GET_INTEGRITY_INFORMATION_BUFFER*;
-typedef struct _FSCTL_SET_INTEGRITY_INFORMATION_BUFFER {
+struct _FSCTL_SET_INTEGRITY_INFORMATION_BUFFER {
WORD ChecksumAlgorithm; // Checksum algorithm. e.g. CHECKSUM_TYPE_UNCHANGED, CHECKSUM_TYPE_NONE, CHECKSUM_TYPE_CRC32
WORD Reserved; // Must be 0
DWORD Flags; // FSCTL_INTEGRITY_FLAG_xxx
-} FSCTL_SET_INTEGRITY_INFORMATION_BUFFER, *PFSCTL_SET_INTEGRITY_INFORMATION_BUFFER;
+};
+
+using FSCTL_SET_INTEGRITY_INFORMATION_BUFFER = _FSCTL_SET_INTEGRITY_INFORMATION_BUFFER;
+using PFSCTL_SET_INTEGRITY_INFORMATION_BUFFER = _FSCTL_SET_INTEGRITY_INFORMATION_BUFFER*;
#endif
@@ -194,6 +203,40 @@ void write(const QString& filename, const QByteArray& data)
}
}
+void appendSafe(const QString& filename, const QByteArray& data)
+{
+ ensureExists(QFileInfo(filename).dir());
+ QByteArray buffer;
+ try {
+ buffer = read(filename);
+ } catch (FileSystemException&) {
+ buffer = QByteArray();
+ }
+ buffer.append(data);
+ QSaveFile file(filename);
+ if (!file.open(QSaveFile::WriteOnly)) {
+ throw FileSystemException("Couldn't open " + filename + " for writing: " + file.errorString());
+ }
+ if (buffer.size() != file.write(buffer)) {
+ throw FileSystemException("Error writing data to " + filename + ": " + file.errorString());
+ }
+ if (!file.commit()) {
+ throw FileSystemException("Error while committing data to " + filename + ": " + file.errorString());
+ }
+}
+
+void append(const QString& filename, const QByteArray& data)
+{
+ ensureExists(QFileInfo(filename).dir());
+ QFile file(filename);
+ if (!file.open(QFile::Append)) {
+ throw FileSystemException("Couldn't open " + filename + " for writing: " + file.errorString());
+ }
+ if (data.size() != file.write(data)) {
+ throw FileSystemException("Error writing data to " + filename + ": " + file.errorString());
+ }
+}
+
QByteArray read(const QString& filename)
{
QFile file(filename);
@@ -238,6 +281,28 @@ bool ensureFolderPathExists(QString foldernamepath)
return success;
}
+bool copyFileAttributes(QString src, QString dst)
+{
+#ifdef Q_OS_WIN32
+ auto attrs = GetFileAttributesW(src.toStdWString().c_str());
+ if (attrs == INVALID_FILE_ATTRIBUTES)
+ return false;
+ return SetFileAttributesW(dst.toStdWString().c_str(), attrs);
+#endif
+ return true;
+}
+
+// needs folders to exists
+void copyFolderAttributes(QString src, QString dst, QString relative)
+{
+ auto path = PathCombine(src, relative);
+ QDir dsrc(src);
+ while ((path = QFileInfo(path).path()).length() >= src.length()) {
+ auto dst_path = PathCombine(dst, dsrc.relativeFilePath(path));
+ copyFileAttributes(path, dst_path);
+ }
+}
+
/**
* @brief Copies a directory and it's contents from src to dest
* @param offset subdirectory form src to copy to dest
@@ -265,6 +330,9 @@ bool copy::operator()(const QString& offset, bool dryRun)
if (!m_followSymlinks)
opt |= copy_opts::copy_symlinks;
+ if (m_overwrite)
+ opt |= copy_opts::overwrite_existing;
+
// Function that'll do the actual copying
auto copy_file = [&](QString src_path, QString relative_dst_path) {
if (m_matcher && (m_matcher->matches(relative_dst_path) != m_whitelist))
@@ -273,6 +341,9 @@ bool copy::operator()(const QString& offset, bool dryRun)
auto dst_path = PathCombine(dst, relative_dst_path);
if (!dryRun) {
ensureFilePathExists(dst_path);
+#ifdef Q_OS_WIN32
+ copyFolderAttributes(src, dst, relative_dst_path);
+#endif
fs::copy(StringUtils::toStdString(src_path), StringUtils::toStdString(dst_path), opt, err);
}
if (err) {
diff --git a/launcher/FileSystem.h b/launcher/FileSystem.h
index bfed576c1..861cfa267 100644
--- a/launcher/FileSystem.h
+++ b/launcher/FileSystem.h
@@ -61,6 +61,16 @@ class FileSystemException : public ::Exception {
*/
void write(const QString& filename, const QByteArray& data);
+/**
+ * append data to a file safely
+ */
+void appendSafe(const QString& filename, const QByteArray& data);
+
+/**
+ * append data to a file
+ */
+void append(const QString& filename, const QByteArray& data);
+
/**
* read data from a file safely\
*/
@@ -109,6 +119,11 @@ class copy : public QObject {
m_whitelist = whitelist;
return *this;
}
+ copy& overwrite(const bool overwrite)
+ {
+ m_overwrite = overwrite;
+ return *this;
+ }
bool operator()(bool dryRun = false) { return operator()(QString(), dryRun); }
@@ -128,6 +143,7 @@ class copy : public QObject {
bool m_followSymlinks = true;
const IPathMatcher* m_matcher = nullptr;
bool m_whitelist = false;
+ bool m_overwrite = false;
QDir m_src;
QDir m_dst;
qsizetype m_copied;
diff --git a/launcher/InstanceList.cpp b/launcher/InstanceList.cpp
index 856eee816..2b8f34293 100644
--- a/launcher/InstanceList.cpp
+++ b/launcher/InstanceList.cpp
@@ -2,6 +2,7 @@
/*
* Prism Launcher - Minecraft Launcher
* Copyright (C) 2022 Sefa Eyeoglu
+ * Copyright (C) 2023 TheKodeToad
*
* 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
@@ -237,8 +238,11 @@ GroupId InstanceList::getInstanceGroup(const InstanceId& id) const
return GroupId();
}
-void InstanceList::setInstanceGroup(const InstanceId& id, const GroupId& name)
+void InstanceList::setInstanceGroup(const InstanceId& id, GroupId name)
{
+ if (name.isEmpty() && !name.isNull())
+ name = QString();
+
auto inst = getInstanceById(id);
if (!inst) {
qDebug() << "Attempt to set a null instance's group";
@@ -249,6 +253,7 @@ void InstanceList::setInstanceGroup(const InstanceId& id, const GroupId& name)
auto iter = m_instanceGroupIndex.find(inst->id());
if (iter != m_instanceGroupIndex.end()) {
if (*iter != name) {
+ decreaseGroupCount(*iter);
*iter = name;
changed = true;
}
@@ -258,7 +263,7 @@ void InstanceList::setInstanceGroup(const InstanceId& id, const GroupId& name)
}
if (changed) {
- m_groupNameCache.insert(name);
+ increaseGroupCount(name);
auto idx = getInstIndex(inst.get());
emit dataChanged(index(idx), index(idx), { GroupRole });
saveGroupList();
@@ -267,29 +272,55 @@ void InstanceList::setInstanceGroup(const InstanceId& id, const GroupId& name)
QStringList InstanceList::getGroups()
{
- return m_groupNameCache.values();
+ return m_groupNameCache.keys();
}
-void InstanceList::deleteGroup(const QString& name)
+void InstanceList::deleteGroup(const GroupId& name)
{
+ m_groupNameCache.remove(name);
+ m_collapsedGroups.remove(name);
+
bool removed = false;
qDebug() << "Delete group" << name;
for (auto& instance : m_instances) {
- const auto& instID = instance->id();
- auto instGroupName = getInstanceGroup(instID);
+ const QString& instID = instance->id();
+ const QString instGroupName = getInstanceGroup(instID);
if (instGroupName == name) {
m_instanceGroupIndex.remove(instID);
qDebug() << "Remove" << instID << "from group" << name;
removed = true;
auto idx = getInstIndex(instance.get());
- if (idx > 0) {
+ if (idx >= 0)
emit dataChanged(index(idx), index(idx), { GroupRole });
- }
}
}
- if (removed) {
+ if (removed)
saveGroupList();
+}
+
+void InstanceList::renameGroup(const QString& src, const QString& dst)
+{
+ m_groupNameCache.remove(src);
+ if (m_collapsedGroups.remove(src))
+ m_collapsedGroups.insert(dst);
+
+ bool modified = false;
+ qDebug() << "Rename group" << src << "to" << dst;
+ for (auto& instance : m_instances) {
+ const QString& instID = instance->id();
+ const QString instGroupName = getInstanceGroup(instID);
+ if (instGroupName == src) {
+ m_instanceGroupIndex[instID] = dst;
+ increaseGroupCount(dst);
+ qDebug() << "Set" << instID << "group to" << dst;
+ modified = true;
+ auto idx = getInstIndex(instance.get());
+ if (idx >= 0)
+ emit dataChanged(index(idx), index(idx), { GroupRole });
+ }
}
+ if (modified)
+ saveGroupList();
}
bool InstanceList::isGroupCollapsed(const QString& group)
@@ -305,12 +336,13 @@ bool InstanceList::trashInstance(const InstanceId& id)
return false;
}
- auto cachedGroupId = m_instanceGroupIndex[id];
+ QString cachedGroupId = m_instanceGroupIndex[id];
qDebug() << "Will trash instance" << id;
QString trashedLoc;
if (m_instanceGroupIndex.remove(id)) {
+ decreaseGroupCount(cachedGroupId);
saveGroupList();
}
@@ -348,7 +380,7 @@ void InstanceList::undoTrashInstance()
QFile(top.trashPath).rename(top.polyPath);
m_instanceGroupIndex[top.id] = top.groupName;
- m_groupNameCache.insert(top.groupName);
+ increaseGroupCount(top.groupName);
saveGroupList();
emit instancesChanged();
@@ -362,7 +394,10 @@ void InstanceList::deleteInstance(const InstanceId& id)
return;
}
+ QString cachedGroupId = m_instanceGroupIndex[id];
+
if (m_instanceGroupIndex.remove(id)) {
+ decreaseGroupCount(cachedGroupId);
saveGroupList();
}
@@ -610,6 +645,25 @@ InstancePtr InstanceList::loadInstance(const InstanceId& id)
return inst;
}
+void InstanceList::increaseGroupCount(const QString& group)
+{
+ if (group.isEmpty())
+ return;
+
+ ++m_groupNameCache[group];
+}
+
+void InstanceList::decreaseGroupCount(const QString& group)
+{
+ if (group.isEmpty())
+ return;
+
+ if (--m_groupNameCache[group] < 1) {
+ m_groupNameCache.remove(group);
+ m_collapsedGroups.remove(group);
+ }
+}
+
void InstanceList::saveGroupList()
{
qDebug() << "Will save group list now.";
@@ -621,7 +675,7 @@ void InstanceList::saveGroupList()
QString groupFileName = m_instDir + "/instgroups.json";
QMap> reverseGroupMap;
for (auto iter = m_instanceGroupIndex.begin(); iter != m_instanceGroupIndex.end(); iter++) {
- QString id = iter.key();
+ const QString& id = iter.key();
QString group = iter.value();
if (group.isEmpty())
continue;
@@ -711,17 +765,22 @@ void InstanceList::loadGroupList()
return;
}
- QSet groupSet;
m_instanceGroupIndex.clear();
+ m_groupNameCache.clear();
// Iterate through all the groups.
QJsonObject groupMapping = rootObj.value("groups").toObject();
for (QJsonObject::iterator iter = groupMapping.begin(); iter != groupMapping.end(); iter++) {
QString groupName = iter.key();
+ if (iter.key().isEmpty()) {
+ qWarning() << "Redundant empty group found";
+ continue;
+ }
+
// If not an object, complain and skip to the next one.
if (!iter.value().isObject()) {
- qWarning() << QString("Group '%1' in the group list should be an object.").arg(groupName).toUtf8();
+ qWarning() << QString("Group '%1' in the group list should be an object").arg(groupName).toUtf8();
continue;
}
@@ -733,23 +792,19 @@ void InstanceList::loadGroupList()
continue;
}
- // keep a list/set of groups for choosing
- groupSet.insert(groupName);
-
auto hidden = groupObj.value("hidden").toBool(false);
- if (hidden) {
+ if (hidden)
m_collapsedGroups.insert(groupName);
- }
// Iterate through the list of instances in the group.
QJsonArray instancesArray = groupObj.value("instances").toArray();
- for (QJsonArray::iterator iter2 = instancesArray.begin(); iter2 != instancesArray.end(); iter2++) {
- m_instanceGroupIndex[(*iter2).toString()] = groupName;
+ for (auto value : instancesArray) {
+ m_instanceGroupIndex[value.toString()] = groupName;
+ increaseGroupCount(groupName);
}
}
m_groupsLoaded = true;
- m_groupNameCache.unite(groupSet);
qDebug() << "Group list loaded.";
}
@@ -892,9 +947,12 @@ QString InstanceList::getStagedInstancePath()
bool InstanceList::commitStagedInstance(const QString& path,
InstanceName const& instanceName,
- const QString& groupName,
+ QString groupName,
InstanceTask const& commiting)
{
+ if (groupName.isEmpty() && !groupName.isNull())
+ groupName = QString();
+
QDir dir;
QString instID;
InstancePtr inst;
@@ -925,7 +983,7 @@ bool InstanceList::commitStagedInstance(const QString& path,
}
m_instanceGroupIndex[instID] = groupName;
- m_groupNameCache.insert(groupName);
+ increaseGroupCount(groupName);
}
instanceSet.insert(instID);
diff --git a/launcher/InstanceList.h b/launcher/InstanceList.h
index ee4578ffd..5ddddee95 100644
--- a/launcher/InstanceList.h
+++ b/launcher/InstanceList.h
@@ -1,16 +1,36 @@
-/* Copyright 2013-2021 MultiMC Contributors
+// SPDX-License-Identifier: GPL-3.0-only
+/*
+ * Prism Launcher - Minecraft Launcher
+ * Copyright (C) 2023 TheKodeToad
*
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
+ * 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.
*
- * http://www.apache.org/licenses/LICENSE-2.0
+ * 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.
*
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ *
+ * 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.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
*/
#pragma once
@@ -86,9 +106,10 @@ class InstanceList : public QAbstractListModel {
bool isGroupCollapsed(const QString& groupName);
GroupId getInstanceGroup(const InstanceId& id) const;
- void setInstanceGroup(const InstanceId& id, const GroupId& name);
+ void setInstanceGroup(const InstanceId& id, GroupId name);
void deleteGroup(const GroupId& name);
+ void renameGroup(const GroupId& src, const GroupId& dst);
bool trashInstance(const InstanceId& id);
bool trashedSomething();
void undoTrashInstance();
@@ -109,7 +130,7 @@ class InstanceList : public QAbstractListModel {
* should_override is used when another similar instance already exists, and we want to override it
* - for instance, when updating it.
*/
- bool commitStagedInstance(const QString& keyPath, const InstanceName& instanceName, const QString& groupName, const InstanceTask&);
+ bool commitStagedInstance(const QString& keyPath, const InstanceName& instanceName, QString groupName, const InstanceTask&);
/**
* Destroy a previously created staging area given by @keyPath - used when creation fails.
@@ -158,12 +179,16 @@ class InstanceList : public QAbstractListModel {
QList discoverInstances();
InstancePtr loadInstance(const InstanceId& id);
+ void increaseGroupCount(const QString& group);
+ void decreaseGroupCount(const QString& group);
+
private:
int m_watchLevel = 0;
int totalPlayTime = 0;
bool m_dirty = false;
QList m_instances;
- QSet m_groupNameCache;
+ // id -> refs
+ QMap m_groupNameCache;
SettingsObjectPtr m_globalSettings;
QString m_instDir;
diff --git a/launcher/LaunchController.cpp b/launcher/LaunchController.cpp
index 1b2e73a62..9956cd2c7 100644
--- a/launcher/LaunchController.cpp
+++ b/launcher/LaunchController.cpp
@@ -93,8 +93,8 @@ void LaunchController::decideAccount()
if (accounts->count() <= 0) {
// Tell the user they need to log in at least one account in order to play.
auto reply = CustomMessageBox::selectable(m_parentWidget, tr("No Accounts"),
- tr("In order to play Minecraft, you must have at least one Microsoft or Mojang "
- "account logged in. Mojang accounts can only be used offline. "
+ tr("In order to play Minecraft, you must have at least one Microsoft "
+ "account which owns Minecraft logged in. "
"Would you like to open the account manager to add an account now?"),
QMessageBox::Information, QMessageBox::Yes | QMessageBox::No)
->exec();
@@ -111,7 +111,7 @@ void LaunchController::decideAccount()
// Select the account to use. If the instance has a specific account set, that will be used. Otherwise, the default account will be used
auto instanceAccountId = m_instance->settings()->get("InstanceAccountId").toString();
auto instanceAccountIndex = accounts->findAccountByProfileId(instanceAccountId);
- if (instanceAccountIndex == -1) {
+ if (instanceAccountIndex == -1 || instanceAccountId.isEmpty()) {
m_accountToUse = accounts->defaultAccount();
} else {
m_accountToUse = accounts->at(instanceAccountIndex);
diff --git a/launcher/MMCZip.cpp b/launcher/MMCZip.cpp
index acd6bf7e4..3bfe16ab5 100644
--- a/launcher/MMCZip.cpp
+++ b/launcher/MMCZip.cpp
@@ -42,7 +42,11 @@
#include
#include
+#include
+
+#if defined(LAUNCHER_APPLICATION)
#include
+#endif
namespace MMCZip {
// ours
@@ -132,6 +136,7 @@ bool compressDirFiles(QString fileCompressed, QString dir, QFileInfoList files,
return result;
}
+#if defined(LAUNCHER_APPLICATION)
// ours
bool createModdedJar(QString sourceJarPath, QString targetJarPath, const QList& mods)
{
@@ -217,6 +222,7 @@ bool createModdedJar(QString sourceJarPath, QString targetJarPath, const QList
#include
#include
+
+#if defined(LAUNCHER_APPLICATION)
#include "minecraft/mod/Mod.h"
+#endif
#include "tasks/Task.h"
namespace MMCZip {
@@ -79,11 +82,12 @@ bool compressDirFiles(QuaZip* zip, QString dir, QFileInfoList files, bool follow
*/
bool compressDirFiles(QString fileCompressed, QString dir, QFileInfoList files, bool followSymlinks = false);
+#if defined(LAUNCHER_APPLICATION)
/**
* take a source jar, add mods to it, resulting in target jar
*/
bool createModdedJar(QString sourceJarPath, QString targetJarPath, const QList& mods);
-
+#endif
/**
* Find a single file in archive by file name (not path)
*
@@ -147,6 +151,7 @@ bool extractFile(QString fileCompressed, QString file, QString dir);
*/
bool collectFileListRecursively(const QString& rootDir, const QString& subDir, QFileInfoList* files, FilterFunction excludeFilter);
+#if defined(LAUNCHER_APPLICATION)
class ExportToZipTask : public Task {
public:
ExportToZipTask(QString outputPath, QDir dir, QFileInfoList files, QString destinationPrefix = "", bool followSymlinks = false)
@@ -167,7 +172,7 @@ class ExportToZipTask : public Task {
void setExcludeFiles(QStringList excludeFiles) { m_exclude_files = excludeFiles; }
void addExtraFile(QString fileName, QByteArray data) { m_extra_files.insert(fileName, data); }
- typedef std::optional ZipResult;
+ using ZipResult = std::optional;
protected:
virtual void executeTask() override;
@@ -189,4 +194,5 @@ class ExportToZipTask : public Task {
QFuture m_build_zip_future;
QFutureWatcher m_build_zip_watcher;
};
+#endif
} // namespace MMCZip
diff --git a/launcher/MTPixmapCache.h b/launcher/MTPixmapCache.h
index 1a3e52160..b6bd13045 100644
--- a/launcher/MTPixmapCache.h
+++ b/launcher/MTPixmapCache.h
@@ -5,6 +5,7 @@
#include
#include
#include
+#include
#define GET_TYPE() \
Qt::ConnectionType type; \
@@ -100,10 +101,14 @@ class PixmapCache final : public QObject {
*/
bool _markCacheMissByEviciton()
{
+ static constexpr uint maxInt = static_cast(std::numeric_limits::max());
+ static constexpr uint step = 10240;
+ static constexpr int oneSecond = 1000;
+
auto now = QTime::currentTime();
if (!m_last_cache_miss_by_eviciton.isNull()) {
auto diff = m_last_cache_miss_by_eviciton.msecsTo(now);
- if (diff < 1000) { // less than a second ago
+ if (diff < oneSecond) { // less than a second ago
++m_consecutive_fast_evicitons;
} else {
m_consecutive_fast_evicitons = 0;
@@ -111,11 +116,17 @@ class PixmapCache final : public QObject {
}
m_last_cache_miss_by_eviciton = now;
if (m_consecutive_fast_evicitons >= m_consecutive_fast_evicitons_threshold) {
- // double the cache size
- auto newSize = _cacheLimit() * 2;
- qDebug() << m_consecutive_fast_evicitons << "pixmap cache misses by eviction happened too fast, doubling cache size to"
- << newSize;
- _setCacheLimit(newSize);
+ // increase the cache size
+ uint newSize = _cacheLimit() + step;
+ if (newSize >= maxInt) { // increase it until you overflow :D
+ newSize = maxInt;
+ qDebug() << m_consecutive_fast_evicitons
+ << tr("pixmap cache misses by eviction happened too fast, doing nothing as the cache size reached it's limit");
+ } else {
+ qDebug() << m_consecutive_fast_evicitons
+ << tr("pixmap cache misses by eviction happened too fast, increasing cache size to") << static_cast(newSize);
+ }
+ _setCacheLimit(static_cast(newSize));
m_consecutive_fast_evicitons = 0;
return true;
}
diff --git a/launcher/StringUtils.cpp b/launcher/StringUtils.cpp
index e08e6fdce..72ccdfbff 100644
--- a/launcher/StringUtils.cpp
+++ b/launcher/StringUtils.cpp
@@ -35,6 +35,7 @@
*/
#include "StringUtils.h"
+#include
#include
#include
@@ -149,7 +150,7 @@ QString StringUtils::truncateUrlHumanFriendly(QUrl& url, int max_len, bool hard_
}
if ((url_compact.length() >= max_len) && hard_limit) {
- // still too long, truncate normaly
+ // still too long, truncate normally
url_compact = QString(str_url);
auto to_remove = url_compact.length() - max_len + 3;
url_compact.remove(url_compact.length() - to_remove - 1, to_remove);
@@ -182,3 +183,32 @@ QString StringUtils::getRandomAlphaNumeric()
{
return QUuid::createUuid().toString(QUuid::Id128);
}
+
+QPair StringUtils::splitFirst(const QString& s, const QString& sep, Qt::CaseSensitivity cs)
+{
+ QString left, right;
+ auto index = s.indexOf(sep, 0, cs);
+ left = s.mid(0, index);
+ right = s.mid(index + sep.length());
+ return qMakePair(left, right);
+}
+
+QPair StringUtils::splitFirst(const QString& s, QChar sep, Qt::CaseSensitivity cs)
+{
+ QString left, right;
+ auto index = s.indexOf(sep, 0, cs);
+ left = s.mid(0, index);
+ right = s.mid(left.length() + 1);
+ return qMakePair(left, right);
+}
+
+QPair StringUtils::splitFirst(const QString& s, const QRegularExpression& re)
+{
+ QString left, right;
+ QRegularExpressionMatch match;
+ auto index = s.indexOf(re, 0, &match);
+ left = s.mid(0, index);
+ auto end = match.hasMatch() ? left.length() + match.capturedLength() : left.length() + 1;
+ right = s.mid(end);
+ return qMakePair(left, right);
+}
diff --git a/launcher/StringUtils.h b/launcher/StringUtils.h
index eac0d5a7d..9d2bdd85e 100644
--- a/launcher/StringUtils.h
+++ b/launcher/StringUtils.h
@@ -36,8 +36,10 @@
#pragma once
+#include
#include
#include
+#include
namespace StringUtils {
@@ -70,12 +72,17 @@ int naturalCompare(const QString& s1, const QString& s2, Qt::CaseSensitivity cs)
/**
* @brief Truncate a url while keeping its readability py placing the `...` in the middle of the path
* @param url Url to truncate
- * @param max_len max lenght of url in charaters
- * @param hard_limit if truncating the path can't get the url short enough, truncate it normaly.
+ * @param max_len max length of url in characters
+ * @param hard_limit if truncating the path can't get the url short enough, truncate it normally.
*/
QString truncateUrlHumanFriendly(QUrl& url, int max_len, bool hard_limit = false);
QString humanReadableFileSize(double bytes, bool use_si = false, int decimal_points = 1);
QString getRandomAlphaNumeric();
+
+QPair splitFirst(const QString& s, const QString& sep, Qt::CaseSensitivity cs = Qt::CaseSensitive);
+QPair splitFirst(const QString& s, QChar sep, Qt::CaseSensitivity cs = Qt::CaseSensitive);
+QPair splitFirst(const QString& s, const QRegularExpression& re);
+
} // namespace StringUtils
diff --git a/launcher/Version.h b/launcher/Version.h
index 9c043b013..b06e256aa 100644
--- a/launcher/Version.h
+++ b/launcher/Version.h
@@ -56,6 +56,7 @@ class Version {
bool operator!=(const Version& other) const;
QString toString() const { return m_string; }
+ bool isEmpty() const { return m_string.isEmpty(); }
friend QDebug operator<<(QDebug debug, const Version& v);
diff --git a/launcher/VersionProxyModel.h b/launcher/VersionProxyModel.h
index c7d5fd94e..57c711d58 100644
--- a/launcher/VersionProxyModel.h
+++ b/launcher/VersionProxyModel.h
@@ -10,7 +10,7 @@ class VersionProxyModel : public QAbstractProxyModel {
Q_OBJECT
public:
enum Column { Name, ParentVersion, Branch, Type, Architecture, Path, Time };
- typedef QHash> FilterMap;
+ using FilterMap = QHash>;
public:
VersionProxyModel(QObject* parent = 0);
diff --git a/launcher/filelink/FileLink.cpp b/launcher/filelink/FileLink.cpp
index 1ea0e382f..bdf173ebc 100644
--- a/launcher/filelink/FileLink.cpp
+++ b/launcher/filelink/FileLink.cpp
@@ -93,6 +93,7 @@ FileLinkApp::FileLinkApp(int& argc, char** argv) : QCoreApplication(argc, argv),
joinServer(serverToJoin);
} else {
qDebug() << "no server to join";
+ m_status = Failed;
exit();
}
}
@@ -108,6 +109,7 @@ void FileLinkApp::joinServer(QString server)
connect(&socket, &QLocalSocket::readyRead, this, &FileLinkApp::readPathPairs);
connect(&socket, &QLocalSocket::errorOccurred, this, [&](QLocalSocket::LocalSocketError socketError) {
+ m_status = Failed;
switch (socketError) {
case QLocalSocket::ServerNotFoundError:
qDebug()
@@ -132,6 +134,7 @@ void FileLinkApp::joinServer(QString server)
connect(&socket, &QLocalSocket::disconnected, this, [&]() {
qDebug() << "disconnected from server, should exit";
+ m_status = Succeeded;
exit();
});
diff --git a/launcher/filelink/FileLink.h b/launcher/filelink/FileLink.h
index 4c47d9bbb..583d0d43a 100644
--- a/launcher/filelink/FileLink.h
+++ b/launcher/filelink/FileLink.h
@@ -41,8 +41,10 @@ class FileLinkApp : public QCoreApplication {
// friends for the purpose of limiting access to deprecated stuff
Q_OBJECT
public:
+ enum Status { Starting, Failed, Succeeded, Initialized };
FileLinkApp(int& argc, char** argv);
virtual ~FileLinkApp();
+ Status status() const { return m_status; }
private:
void joinServer(QString server);
@@ -50,6 +52,8 @@ class FileLinkApp : public QCoreApplication {
void runLink();
void sendResults();
+ Status m_status = Status::Starting;
+
bool m_useHardLinks = false;
QDateTime m_startTime;
diff --git a/launcher/filelink/main.cpp b/launcher/filelink/filelink_main.cpp
similarity index 75%
rename from launcher/filelink/main.cpp
rename to launcher/filelink/filelink_main.cpp
index 83566a3c6..2a8bcb703 100644
--- a/launcher/filelink/main.cpp
+++ b/launcher/filelink/filelink_main.cpp
@@ -26,5 +26,16 @@ int main(int argc, char* argv[])
{
FileLinkApp ldh(argc, argv);
- return ldh.exec();
+ switch (ldh.status()) {
+ case FileLinkApp::Starting:
+ case FileLinkApp::Initialized: {
+ return ldh.exec();
+ }
+ case FileLinkApp::Failed:
+ return 1;
+ case FileLinkApp::Succeeded:
+ return 0;
+ default:
+ return -1;
+ }
}
diff --git a/launcher/java/JavaChecker.h b/launcher/java/JavaChecker.h
index 743f49417..7111f8522 100644
--- a/launcher/java/JavaChecker.h
+++ b/launcher/java/JavaChecker.h
@@ -22,8 +22,8 @@ struct JavaCheckResult {
enum class Validity { Errored, ReturnedInvalidData, Valid } validity = Validity::Errored;
};
-typedef shared_qobject_ptr QProcessPtr;
-typedef shared_qobject_ptr JavaCheckerPtr;
+using QProcessPtr = shared_qobject_ptr;
+using JavaCheckerPtr = shared_qobject_ptr;
class JavaChecker : public QObject {
Q_OBJECT
public:
diff --git a/launcher/java/JavaCheckerJob.h b/launcher/java/JavaCheckerJob.h
index 009687917..ddf827968 100644
--- a/launcher/java/JavaCheckerJob.h
+++ b/launcher/java/JavaCheckerJob.h
@@ -20,7 +20,7 @@
#include "tasks/Task.h"
class JavaCheckerJob;
-typedef shared_qobject_ptr JavaCheckerJobPtr;
+using JavaCheckerJobPtr = shared_qobject_ptr;
// FIXME: this just seems horribly redundant
class JavaCheckerJob : public Task {
diff --git a/launcher/java/JavaInstall.h b/launcher/java/JavaInstall.h
index 30815b5a8..8c2743a00 100644
--- a/launcher/java/JavaInstall.h
+++ b/launcher/java/JavaInstall.h
@@ -24,11 +24,11 @@
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 descriptor() override { return id.toString(); }
- virtual QString name() { return id.toString(); }
+ virtual QString name() override { return id.toString(); }
- virtual QString typeString() const { return arch; }
+ virtual QString typeString() const override { return arch; }
virtual bool operator<(BaseVersion& a) override;
virtual bool operator>(BaseVersion& a) override;
@@ -42,4 +42,4 @@ struct JavaInstall : public BaseVersion {
bool recommended = false;
};
-typedef std::shared_ptr JavaInstallPtr;
+using JavaInstallPtr = std::shared_ptr;
diff --git a/launcher/java/JavaUtils.cpp b/launcher/java/JavaUtils.cpp
index 43624a1e2..0612c02b0 100644
--- a/launcher/java/JavaUtils.cpp
+++ b/launcher/java/JavaUtils.cpp
@@ -335,6 +335,7 @@ QList JavaUtils::FindJavaPaths()
}
}
+ candidates.append(getMinecraftJavaBundle());
candidates = addJavasFromEnv(candidates);
candidates.removeDuplicates();
return candidates;
@@ -360,6 +361,7 @@ QList JavaUtils::FindJavaPaths()
javas.append(systemLibraryJVMDir.absolutePath() + "/" + java + "/Contents/Home/bin/java");
javas.append(systemLibraryJVMDir.absolutePath() + "/" + java + "/Contents/Commands/java");
}
+ javas.append(getMinecraftJavaBundle());
javas = addJavasFromEnv(javas);
javas.removeDuplicates();
return javas;
@@ -403,6 +405,15 @@ QList JavaUtils::FindJavaPaths()
scanJavaDirs("/opt/jdks");
// flatpak
scanJavaDirs("/app/jdk");
+
+ auto home = qEnvironmentVariable("HOME");
+
+ // javas downloaded by IntelliJ
+ scanJavaDirs(FS::PathCombine(home, ".jdks"));
+ // javas downloaded by sdkman
+ scanJavaDirs(FS::PathCombine(home, ".sdkman/candidates/java"));
+
+ javas.append(getMinecraftJavaBundle());
javas = addJavasFromEnv(javas);
javas.removeDuplicates();
return javas;
@@ -415,6 +426,7 @@ QList JavaUtils::FindJavaPaths()
QList javas;
javas.append(this->GetDefaultJava()->path);
+ javas.append(getMinecraftJavaBundle());
return addJavasFromEnv(javas);
}
#endif
@@ -423,3 +435,42 @@ QString JavaUtils::getJavaCheckPath()
{
return APPLICATION->getJarPath("JavaCheck.jar");
}
+
+QStringList getMinecraftJavaBundle()
+{
+ QString partialPath;
+ QString executable = "java";
+#if defined(Q_OS_OSX)
+ partialPath = FS::PathCombine(QDir::homePath(), "Library/Application Support");
+#elif defined(Q_OS_WIN32)
+ partialPath = QProcessEnvironment::systemEnvironment().value("LOCALAPPDATA", "");
+ executable += "w.exe";
+#else
+ partialPath = QDir::homePath();
+#endif
+ auto minecraftPath = FS::PathCombine(partialPath, ".minecraft", "runtime");
+ QStringList javas;
+ QStringList processpaths{ minecraftPath };
+
+ while (!processpaths.isEmpty()) {
+ auto dirPath = processpaths.takeFirst();
+ QDir dir(dirPath);
+ if (!dir.exists())
+ continue;
+ auto entries = dir.entryInfoList(QDir::Dirs | QDir::NoDotAndDotDot);
+ auto binFound = false;
+ for (auto& entry : entries) {
+ if (entry.baseName() == "bin") {
+ javas.append(FS::PathCombine(entry.canonicalFilePath(), executable));
+ binFound = true;
+ break;
+ }
+ }
+ if (!binFound) {
+ for (auto& entry : entries) {
+ processpaths << entry.canonicalFilePath();
+ }
+ }
+ }
+ return javas;
+}
diff --git a/launcher/java/JavaUtils.h b/launcher/java/JavaUtils.h
index 616179706..2fb03af7a 100644
--- a/launcher/java/JavaUtils.h
+++ b/launcher/java/JavaUtils.h
@@ -26,6 +26,7 @@
QString stripVariableEntries(QString name, QString target, QString remove);
QProcessEnvironment CleanEnviroment();
+QStringList getMinecraftJavaBundle();
class JavaUtils : public QObject {
Q_OBJECT
diff --git a/launcher/java/JavaVersion.cpp b/launcher/java/JavaVersion.cpp
index f9ac47824..b77bf2adf 100644
--- a/launcher/java/JavaVersion.cpp
+++ b/launcher/java/JavaVersion.cpp
@@ -45,10 +45,12 @@ QString JavaVersion::toString() const
bool JavaVersion::requiresPermGen()
{
- if (m_parseable) {
- return m_major < 8;
- }
- return true;
+ return !m_parseable || m_major < 8;
+}
+
+bool JavaVersion::isModular()
+{
+ return m_parseable && m_major >= 9;
}
bool JavaVersion::operator<(const JavaVersion& rhs)
diff --git a/launcher/java/JavaVersion.h b/launcher/java/JavaVersion.h
index 7e66269cb..421578ea1 100644
--- a/launcher/java/JavaVersion.h
+++ b/launcher/java/JavaVersion.h
@@ -25,6 +25,8 @@ class JavaVersion {
bool requiresPermGen();
+ bool isModular();
+
QString toString() const;
int major() { return m_major; }
diff --git a/launcher/minecraft/Agent.h b/launcher/minecraft/Agent.h
index 8958521e5..bc385a74e 100644
--- a/launcher/minecraft/Agent.h
+++ b/launcher/minecraft/Agent.h
@@ -6,7 +6,7 @@
class Agent;
-typedef std::shared_ptr AgentPtr;
+using AgentPtr = std::shared_ptr;
class Agent {
public:
diff --git a/launcher/minecraft/AssetsUtils.cpp b/launcher/minecraft/AssetsUtils.cpp
index 4ef007075..48e150d16 100644
--- a/launcher/minecraft/AssetsUtils.cpp
+++ b/launcher/minecraft/AssetsUtils.cpp
@@ -48,6 +48,7 @@
#include "FileSystem.h"
#include "net/ApiDownload.h"
#include "net/ChecksumValidator.h"
+#include "net/Download.h"
#include "Application.h"
diff --git a/launcher/minecraft/Component.h b/launcher/minecraft/Component.h
index 3474a22e0..fdb61c45e 100644
--- a/launcher/minecraft/Component.h
+++ b/launcher/minecraft/Component.h
@@ -105,4 +105,4 @@ class Component : public QObject, public ProblemProvider {
bool m_loaded = false;
};
-typedef shared_qobject_ptr ComponentPtr;
+using ComponentPtr = shared_qobject_ptr;
diff --git a/launcher/minecraft/Library.h b/launcher/minecraft/Library.h
index f8816a97b..adb89c4c6 100644
--- a/launcher/minecraft/Library.h
+++ b/launcher/minecraft/Library.h
@@ -52,7 +52,7 @@
class Library;
class MinecraftInstance;
-typedef std::shared_ptr LibraryPtr;
+using LibraryPtr = std::shared_ptr;
class Library {
friend class OneSixVersionFormat;
diff --git a/launcher/minecraft/MinecraftInstance.cpp b/launcher/minecraft/MinecraftInstance.cpp
index b75071511..8cec485ac 100644
--- a/launcher/minecraft/MinecraftInstance.cpp
+++ b/launcher/minecraft/MinecraftInstance.cpp
@@ -184,6 +184,13 @@ void MinecraftInstance::loadSpecificSettings()
m_settings->registerOverride(global_settings->getSetting("CloseAfterLaunch"), miscellaneousOverride);
m_settings->registerOverride(global_settings->getSetting("QuitAfterGameStop"), miscellaneousOverride);
+ // Legacy-related options
+ auto legacySettings = m_settings->registerSetting("OverrideLegacySettings", false);
+ m_settings->registerOverride(global_settings->getSetting("OnlineFixes"), legacySettings);
+
+ auto envSetting = m_settings->registerSetting("OverrideEnv", false);
+ m_settings->registerOverride(global_settings->getSetting("Env"), envSetting);
+
m_settings->set("InstanceType", "OneSix");
}
@@ -311,7 +318,7 @@ QString MinecraftInstance::getLocalLibraryPath() const
bool MinecraftInstance::supportsDemo() const
{
Version instance_ver{ getPackProfile()->getComponentVersion("net.minecraft") };
- // Demo mode was introduced in 1.3.1: https://minecraft.fandom.com/wiki/Demo_mode#History
+ // Demo mode was introduced in 1.3.1: https://minecraft.wiki/w/Demo_mode#History
// FIXME: Due to Version constraints atm, this can't handle well non-release versions
return instance_ver >= Version("1.3.1");
}
@@ -516,15 +523,18 @@ QStringList MinecraftInstance::javaArguments()
args << "-Duser.language=en";
+ if (javaVersion.isModular() && shouldApplyOnlineFixes())
+ // allow reflective access to java.net - required by the skin fix
+ args << "--add-opens"
+ << "java.base/java.net=ALL-UNNAMED";
+
return args;
}
QString MinecraftInstance::getLauncher()
{
- auto profile = m_components->getProfile();
-
// use legacy launcher if the traits are set
- if (profile->getTraits().contains("legacyLaunch") || profile->getTraits().contains("alphaLaunch"))
+ if (traits().contains("legacyLaunch") || traits().contains("alphaLaunch"))
return "legacy";
return "standard";
@@ -559,6 +569,11 @@ QStringList MinecraftInstance::processAuthArgs(AuthSessionPtr session) const
return args;
}
+bool MinecraftInstance::shouldApplyOnlineFixes()
+{
+ return traits().contains("legacyServices") && settings()->get("OnlineFixes").toBool();
+}
+
QMap MinecraftInstance::getVariables()
{
QMap out;
@@ -568,6 +583,7 @@ QMap MinecraftInstance::getVariables()
out.insert("INST_MC_DIR", QDir::toNativeSeparators(QDir(gameRoot()).absolutePath()));
out.insert("INST_JAVA", settings()->get("JavaPath").toString());
out.insert("INST_JAVA_ARGS", javaArguments().join(' '));
+ out.insert("NO_COLOR", "1");
return out;
}
@@ -591,15 +607,20 @@ QProcessEnvironment MinecraftInstance::createLaunchEnvironment()
#ifdef Q_OS_LINUX
if (settings()->get("EnableMangoHud").toBool() && APPLICATION->capabilities() & Application::SupportsMangoHud) {
- auto preloadList = env.value("LD_PRELOAD").split(QLatin1String(":"));
- auto libPaths = env.value("LD_LIBRARY_PATH").split(QLatin1String(":"));
+ QStringList preloadList;
+ if (auto value = env.value("LD_PRELOAD"); !value.isEmpty())
+ preloadList = value.split(QLatin1String(":"));
+ QStringList libPaths;
+ if (auto value = env.value("LD_LIBRARY_PATH"); !value.isEmpty())
+ libPaths = value.split(QLatin1String(":"));
auto mangoHudLibString = MangoHud::getLibraryString();
if (!mangoHudLibString.isEmpty()) {
QFileInfo mangoHudLib(mangoHudLibString);
// dlsym variant is only needed for OpenGL and not included in the vulkan layer
- preloadList << "libMangoHud_dlsym.so" << mangoHudLib.fileName();
+ preloadList << "libMangoHud_dlsym.so"
+ << "libMangoHud_opengl.so" << mangoHudLib.fileName();
libPaths << mangoHudLib.absolutePath();
}
@@ -618,6 +639,23 @@ QProcessEnvironment MinecraftInstance::createLaunchEnvironment()
}
#endif
+ // custom env
+
+ auto insertEnv = [&env](QMap envMap) {
+ if (envMap.isEmpty())
+ return;
+
+ for (auto iter = envMap.begin(); iter != envMap.end(); iter++)
+ env.insert(iter.key(), iter.value().toString());
+ };
+
+ bool overrideEnv = settings()->get("OverrideEnv").toBool();
+
+ if (!overrideEnv)
+ insertEnv(APPLICATION->settings()->get("Env").toMap());
+ else
+ insertEnv(settings()->get("Env").toMap());
+
return env;
}
@@ -730,7 +768,7 @@ QString MinecraftInstance::createLaunchScript(AuthSessionPtr session, MinecraftS
{
QString windowParams;
if (settings()->get("LaunchMaximized").toBool())
- windowParams = "max";
+ windowParams = "maximized";
else
windowParams =
QString("%1x%2").arg(settings()->get("MinecraftWinWidth").toInt()).arg(settings()->get("MinecraftWinHeight").toInt());
@@ -738,6 +776,19 @@ QString MinecraftInstance::createLaunchScript(AuthSessionPtr session, MinecraftS
launchScript += "windowParams " + windowParams + "\n";
}
+ // launcher info
+ {
+ launchScript += "launcherBrand " + BuildConfig.LAUNCHER_NAME + "\n";
+ launchScript += "launcherVersion " + BuildConfig.printableVersionString() + "\n";
+ }
+
+ // instance info
+ {
+ launchScript += "instanceName " + name() + "\n";
+ launchScript += "instanceIconKey " + name() + "\n";
+ launchScript += "instanceIconPath icon.png\n"; // we already save a copy here
+ }
+
// legacy auth
if (session) {
launchScript += "userName " + session->player_name + "\n";
@@ -748,6 +799,9 @@ QString MinecraftInstance::createLaunchScript(AuthSessionPtr session, MinecraftS
launchScript += "traits " + trait + "\n";
}
+ if (shouldApplyOnlineFixes())
+ launchScript += "onlineFixes true\n";
+
launchScript += "launcher " + getLauncher() + "\n";
// qDebug() << "Generated launch script:" << launchScript;
@@ -888,9 +942,6 @@ QMap MinecraftInstance::createCensorFilterFromSession(AuthSess
if (sessionRef.access_token != "0") {
addToFilter(sessionRef.access_token, tr(""));
}
- if (sessionRef.client_token.size()) {
- addToFilter(sessionRef.client_token, tr(""));
- }
addToFilter(sessionRef.uuid, tr(""));
return filter;
diff --git a/launcher/minecraft/MinecraftInstance.h b/launcher/minecraft/MinecraftInstance.h
index c10dd1cfa..3e2e5ce99 100644
--- a/launcher/minecraft/MinecraftInstance.h
+++ b/launcher/minecraft/MinecraftInstance.h
@@ -132,6 +132,7 @@ class MinecraftInstance : public BaseInstance {
/// get arguments passed to java
QStringList javaArguments();
QString getLauncher();
+ bool shouldApplyOnlineFixes();
/// get variables for launch command variable substitution/environment
QMap getVariables() override;
@@ -178,4 +179,4 @@ class MinecraftInstance : public BaseInstance {
mutable std::shared_ptr m_game_options;
};
-typedef std::shared_ptr MinecraftInstancePtr;
+using MinecraftInstancePtr = std::shared_ptr;
diff --git a/launcher/minecraft/MojangDownloadInfo.h b/launcher/minecraft/MojangDownloadInfo.h
index 855dbe005..eb64f95b7 100644
--- a/launcher/minecraft/MojangDownloadInfo.h
+++ b/launcher/minecraft/MojangDownloadInfo.h
@@ -5,7 +5,7 @@
struct MojangDownloadInfo {
// types
- typedef std::shared_ptr Ptr;
+ using Ptr = std::shared_ptr;
// data
/// Local filesystem path. WARNING: not used, only here so we can pass through mojang files unmolested!
@@ -23,7 +23,7 @@ struct MojangLibraryDownloadInfo {
MojangLibraryDownloadInfo() {}
// types
- typedef std::shared_ptr Ptr;
+ using Ptr = std::shared_ptr;
// methods
MojangDownloadInfo* getDownloadInfo(QString classifier)
@@ -42,7 +42,7 @@ struct MojangLibraryDownloadInfo {
struct MojangAssetIndexInfo : public MojangDownloadInfo {
// types
- typedef std::shared_ptr Ptr;
+ using Ptr = std::shared_ptr;
// methods
MojangAssetIndexInfo() {}
diff --git a/launcher/minecraft/PackProfile.cpp b/launcher/minecraft/PackProfile.cpp
index d24ee3bee..8270b3447 100644
--- a/launcher/minecraft/PackProfile.cpp
+++ b/launcher/minecraft/PackProfile.cpp
@@ -1020,8 +1020,7 @@ std::optional PackProfile::getSupportedModLoaders()
// TODO: remove this or add version condition once Quilt drops official Fabric support
if (loaders & ModPlatform::Quilt)
loaders |= ModPlatform::Fabric;
- // TODO: remove this or add version condition once NeoForge drops official Forge support
- if (loaders & ModPlatform::NeoForge)
+ if (getComponentVersion("net.minecraft") == "1.20.1" && (loaders & ModPlatform::NeoForge))
loaders |= ModPlatform::Forge;
return loaders;
}
diff --git a/launcher/minecraft/ProfileUtils.h b/launcher/minecraft/ProfileUtils.h
index 98a7ff739..edabe0bf0 100644
--- a/launcher/minecraft/ProfileUtils.h
+++ b/launcher/minecraft/ProfileUtils.h
@@ -38,7 +38,7 @@
#include "VersionFile.h"
namespace ProfileUtils {
-typedef QStringList PatchOrder;
+using PatchOrder = QStringList;
/// Read and parse a OneSix format order file
bool readOverrideOrders(QString path, PatchOrder& order);
diff --git a/launcher/minecraft/auth/AccountData.cpp b/launcher/minecraft/auth/AccountData.cpp
index ad4b8b436..dda62a6dc 100644
--- a/launcher/minecraft/auth/AccountData.cpp
+++ b/launcher/minecraft/auth/AccountData.cpp
@@ -279,67 +279,6 @@ bool entitlementFromJSONV3(const QJsonObject& parent, MinecraftEntitlement& out)
} // namespace
-bool AccountData::resumeStateFromV2(QJsonObject data)
-{
- // The JSON object must at least have a username for it to be valid.
- if (!data.value("username").isString()) {
- qCritical() << "Can't load Mojang account info from JSON object. Username field is missing or of the wrong type.";
- return false;
- }
-
- QString userName = data.value("username").toString("");
- QString clientToken = data.value("clientToken").toString("");
- QString accessToken = data.value("accessToken").toString("");
-
- QJsonArray profileArray = data.value("profiles").toArray();
- if (profileArray.size() < 1) {
- qCritical() << "Can't load Mojang account with username \"" << userName << "\". No profiles found.";
- return false;
- }
-
- struct AccountProfile {
- QString id;
- QString name;
- bool legacy;
- };
-
- QList profiles;
- int currentProfileIndex = 0;
- int index = -1;
- QString currentProfile = data.value("activeProfile").toString("");
- for (QJsonValue profileVal : profileArray) {
- index++;
- QJsonObject profileObject = profileVal.toObject();
- QString id = profileObject.value("id").toString("");
- QString name = profileObject.value("name").toString("");
- bool legacy_ = profileObject.value("legacy").toBool(false);
- if (id.isEmpty() || name.isEmpty()) {
- qWarning() << "Unable to load a profile" << name << "because it was missing an ID or a name.";
- continue;
- }
- if (id == currentProfile) {
- currentProfileIndex = index;
- }
- profiles.append({ id, name, legacy_ });
- }
- auto& profile = profiles[currentProfileIndex];
-
- type = AccountType::Mojang;
- legacy = profile.legacy;
-
- minecraftProfile.id = profile.id;
- minecraftProfile.name = profile.name;
- minecraftProfile.validity = Katabasis::Validity::Assumed;
-
- yggdrasilToken.token = accessToken;
- yggdrasilToken.extra["clientToken"] = clientToken;
- yggdrasilToken.extra["userName"] = userName;
- yggdrasilToken.validity = Katabasis::Validity::Assumed;
-
- validity_ = minecraftProfile.validity;
- return true;
-}
-
bool AccountData::resumeStateFromV3(QJsonObject data)
{
auto typeV = data.value("type");
@@ -407,15 +346,7 @@ bool AccountData::resumeStateFromV3(QJsonObject data)
QJsonObject AccountData::saveState() const
{
QJsonObject output;
- if (type == AccountType::Mojang) {
- output["type"] = "Mojang";
- if (legacy) {
- output["legacy"] = true;
- }
- if (canMigrateToMSA) {
- output["canMigrateToMSA"] = true;
- }
- } else if (type == AccountType::MSA) {
+ if (type == AccountType::MSA) {
output["type"] = "MSA";
output["msa-client-id"] = msaClientID;
tokenToJSONV3(output, msaToken, "msa");
diff --git a/launcher/minecraft/auth/AccountData.h b/launcher/minecraft/auth/AccountData.h
index ee8be8718..4448025bd 100644
--- a/launcher/minecraft/auth/AccountData.h
+++ b/launcher/minecraft/auth/AccountData.h
@@ -77,7 +77,6 @@ enum class AccountState { Unchecked, Offline, Working, Online, Disabled, Errored
struct AccountData {
QJsonObject saveState() const;
- bool resumeStateFromV2(QJsonObject data);
bool resumeStateFromV3(QJsonObject data);
bool usesCustomApiServers() const;
diff --git a/launcher/minecraft/auth/AccountList.cpp b/launcher/minecraft/auth/AccountList.cpp
index 3101c267d..3e9150554 100644
--- a/launcher/minecraft/auth/AccountList.cpp
+++ b/launcher/minecraft/auth/AccountList.cpp
@@ -54,7 +54,7 @@
#include
-enum AccountListVersion { MojangOnly = 2, MojangMSA = 3 };
+enum AccountListVersion { MojangMSA = 3 };
AccountList::AccountList(QObject* parent) : QAbstractListModel(parent)
{
@@ -379,8 +379,6 @@ QVariant AccountList::headerData(int section, [[maybe_unused]] Qt::Orientation o
return tr("Type");
case StatusColumn:
return tr("Status");
- case MigrationColumn:
- return tr("Can Migrate?");
default:
return QVariant();
}
@@ -392,11 +390,9 @@ QVariant AccountList::headerData(int section, [[maybe_unused]] Qt::Orientation o
case NameColumn:
return tr("User name of the account.");
case TypeColumn:
- return tr("Type of the account - Mojang or MSA.");
+ return tr("Type of the account (MSA or Offline)");
case StatusColumn:
return tr("Current status of the account.");
- case MigrationColumn:
- return tr("Can this account migrate to a Microsoft account?");
default:
return QVariant();
}
@@ -486,9 +482,6 @@ bool AccountList::loadList()
// Make sure the format version matches.
auto listVersion = root.value("formatVersion").toVariant().toInt();
switch (listVersion) {
- case AccountListVersion::MojangOnly: {
- return loadV2(root);
- } break;
case AccountListVersion::MojangMSA: {
return loadV3(root);
} break;
@@ -502,33 +495,6 @@ bool AccountList::loadList()
}
}
-bool AccountList::loadV2(QJsonObject& root)
-{
- beginResetModel();
- auto defaultUserName = root.value("activeAccount").toString("");
- QJsonArray accounts = root.value("accounts").toArray();
- for (QJsonValue accountVal : accounts) {
- QJsonObject accountObj = accountVal.toObject();
- MinecraftAccountPtr account = MinecraftAccount::loadFromJsonV2(accountObj);
- if (account.get() != nullptr) {
- auto profileId = account->profileId();
- if (!profileId.size()) {
- continue;
- }
- connect(account.get(), &MinecraftAccount::changed, this, &AccountList::accountChanged);
- connect(account.get(), &MinecraftAccount::activityChanged, this, &AccountList::accountActivityChanged);
- m_accounts.append(account);
- if (defaultUserName.size() && account->mojangUserName() == defaultUserName) {
- m_defaultAccount = account;
- }
- } else {
- qWarning() << "Failed to load an account.";
- }
- }
- endResetModel();
- return true;
-}
-
bool AccountList::loadV3(QJsonObject& root)
{
beginResetModel();
diff --git a/launcher/minecraft/auth/AccountList.h b/launcher/minecraft/auth/AccountList.h
index ccdf25673..ea301a6c7 100644
--- a/launcher/minecraft/auth/AccountList.h
+++ b/launcher/minecraft/auth/AccountList.h
@@ -98,7 +98,6 @@ class AccountList : public QAbstractListModel {
void setListFilePath(QString path, bool autosave = false);
bool loadList();
- bool loadV2(QJsonObject& root);
bool loadV3(QJsonObject& root);
bool saveList();
diff --git a/launcher/minecraft/auth/AuthSession.h b/launcher/minecraft/auth/AuthSession.h
index 84ecbaac6..489cca24f 100644
--- a/launcher/minecraft/auth/AuthSession.h
+++ b/launcher/minecraft/auth/AuthSession.h
@@ -58,4 +58,4 @@ struct AuthSession {
bool demo = false;
};
-typedef std::shared_ptr AuthSessionPtr;
+using AuthSessionPtr = std::shared_ptr;
diff --git a/launcher/minecraft/auth/MinecraftAccount.cpp b/launcher/minecraft/auth/MinecraftAccount.cpp
index 3a1d5cb36..f419c957d 100644
--- a/launcher/minecraft/auth/MinecraftAccount.cpp
+++ b/launcher/minecraft/auth/MinecraftAccount.cpp
@@ -60,15 +60,6 @@ MinecraftAccount::MinecraftAccount(QObject* parent) : QObject(parent)
data.internalId = QUuid::createUuid().toString(QUuid::Id128);
}
-MinecraftAccountPtr MinecraftAccount::loadFromJsonV2(const QJsonObject& json)
-{
- MinecraftAccountPtr account(new MinecraftAccount());
- if (account->data.resumeStateFromV2(json)) {
- return account;
- }
- return nullptr;
-}
-
MinecraftAccountPtr MinecraftAccount::loadFromJsonV3(const QJsonObject& json)
{
MinecraftAccountPtr account(new MinecraftAccount());
@@ -335,8 +326,6 @@ void MinecraftAccount::fillSession(AuthSessionPtr session)
session->username = data.userName();
// volatile auth token
session->access_token = data.accessToken();
- // the semi-permanent client token
- session->client_token = data.clientToken();
// profile name
session->player_name = data.profileName();
// profile ID
diff --git a/launcher/minecraft/auth/MinecraftAccount.h b/launcher/minecraft/auth/MinecraftAccount.h
index 5ac7ca0f2..1316bb2e2 100644
--- a/launcher/minecraft/auth/MinecraftAccount.h
+++ b/launcher/minecraft/auth/MinecraftAccount.h
@@ -54,7 +54,7 @@ class Task;
class AccountTask;
class MinecraftAccount;
-typedef shared_qobject_ptr MinecraftAccountPtr;
+using MinecraftAccountPtr = shared_qobject_ptr;
Q_DECLARE_METATYPE(MinecraftAccountPtr)
/**
@@ -92,7 +92,6 @@ class MinecraftAccount : public QObject, public Usable {
static MinecraftAccountPtr createOffline(const QString& username);
- static MinecraftAccountPtr loadFromJsonV2(const QJsonObject& json);
static MinecraftAccountPtr loadFromJsonV3(const QJsonObject& json);
static QUuid uuidFromUsername(QString username);
@@ -134,8 +133,6 @@ class MinecraftAccount : public QObject, public Usable {
QString accountDisplayString() const { return data.accountDisplayString(); }
- QString mojangUserName() const { return data.userName(); }
-
QString accessToken() const { return data.accessToken(); }
QString profileId() const { return data.profileId(); }
@@ -147,7 +144,7 @@ class MinecraftAccount : public QObject, public Usable {
bool canMigrate() const { return data.canMigrateToMSA; }
bool isMojangOrMSA() const { return data.type == AccountType::Mojang || data.type == AccountType::MSA; }
-
+
bool isMojang() const { return data.type == AccountType::Mojang; }
bool isMSA() const { return data.type == AccountType::MSA; }
diff --git a/launcher/minecraft/auth/flows/AuthFlow.h b/launcher/minecraft/auth/flows/AuthFlow.h
index c2c412abc..e39e926dd 100644
--- a/launcher/minecraft/auth/flows/AuthFlow.h
+++ b/launcher/minecraft/auth/flows/AuthFlow.h
@@ -12,7 +12,6 @@
#include "minecraft/auth/AccountData.h"
#include "minecraft/auth/AccountTask.h"
#include "minecraft/auth/AuthStep.h"
-#include "minecraft/auth/Yggdrasil.h"
class AuthFlow : public AccountTask {
Q_OBJECT
diff --git a/launcher/minecraft/auth/steps/MinecraftProfileStep.cpp b/launcher/minecraft/auth/steps/MinecraftProfileStep.cpp
index 7cdce23f0..a854342bc 100644
--- a/launcher/minecraft/auth/steps/MinecraftProfileStep.cpp
+++ b/launcher/minecraft/auth/steps/MinecraftProfileStep.cpp
@@ -41,10 +41,6 @@ void MinecraftProfileStep::onRequestDone(QNetworkReply::NetworkError error, QByt
qCDebug(authCredentials()) << data;
if (error == QNetworkReply::ContentNotFoundError) {
// NOTE: Succeed even if we do not have a profile. This is a valid account state.
- if (m_data->type == AccountType::Mojang) {
- m_data->minecraftEntitlement.canPlayMinecraft = false;
- m_data->minecraftEntitlement.ownsMinecraft = false;
- }
m_data->minecraftProfile = MinecraftProfile();
emit finished(AccountTaskState::STATE_SUCCEEDED, tr("Account has no Minecraft profile."));
return;
@@ -73,10 +69,5 @@ void MinecraftProfileStep::onRequestDone(QNetworkReply::NetworkError error, QByt
return;
}
- if (m_data->type == AccountType::Mojang) {
- auto validProfile = m_data->minecraftProfile.validity == Katabasis::Validity::Certain;
- m_data->minecraftEntitlement.canPlayMinecraft = validProfile;
- m_data->minecraftEntitlement.ownsMinecraft = validProfile;
- }
emit finished(AccountTaskState::STATE_WORKING, tr("Minecraft Java profile acquisition succeeded."));
}
diff --git a/launcher/minecraft/auth/steps/XboxUserStep.cpp b/launcher/minecraft/auth/steps/XboxUserStep.cpp
index 61c33a18d..856036d23 100644
--- a/launcher/minecraft/auth/steps/XboxUserStep.cpp
+++ b/launcher/minecraft/auth/steps/XboxUserStep.cpp
@@ -38,7 +38,7 @@ void XboxUserStep::perform()
QNetworkRequest request = QNetworkRequest(QUrl("https://user.auth.xboxlive.com/user/authenticate"));
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json");
request.setRawHeader("Accept", "application/json");
- // set contract-verison header (prevent err 400 bad-request?)
+ // set contract-version header (prevent err 400 bad-request?)
// https://learn.microsoft.com/en-us/gaming/gdk/_content/gc/reference/live/rest/additional/httpstandardheaders
request.setRawHeader("x-xbl-contract-version", "1");
diff --git a/launcher/minecraft/launch/LauncherPartLaunch.cpp b/launcher/minecraft/launch/LauncherPartLaunch.cpp
index 42eb7e02e..cbd9ead67 100644
--- a/launcher/minecraft/launch/LauncherPartLaunch.cpp
+++ b/launcher/minecraft/launch/LauncherPartLaunch.cpp
@@ -105,6 +105,17 @@ void LauncherPartLaunch::executeTask()
auto instance = m_parent->instance();
std::shared_ptr minecraftInstance = std::dynamic_pointer_cast(instance);
+ QString legacyJarPath;
+ if (minecraftInstance->getLauncher() == "legacy" || minecraftInstance->shouldApplyOnlineFixes()) {
+ legacyJarPath = APPLICATION->getJarPath("NewLaunchLegacy.jar");
+ if (legacyJarPath.isEmpty()) {
+ const char* reason = QT_TR_NOOP("Legacy launcher library could not be found. Please check your installation.");
+ emit logLine(tr(reason), MessageLevel::Fatal);
+ emitFailed(tr(reason));
+ return;
+ }
+ }
+
m_launchScript = minecraftInstance->createLaunchScript(m_session, m_serverToJoin);
QStringList args = minecraftInstance->javaArguments();
@@ -123,6 +134,9 @@ void LauncherPartLaunch::executeTask()
auto classPath = minecraftInstance->getClassPath();
classPath.prepend(jarPath);
+ if (!legacyJarPath.isEmpty())
+ classPath.prepend(legacyJarPath);
+
auto natPath = minecraftInstance->getNativePath();
#ifdef Q_OS_WIN
if (!fitsInLocal8bit(natPath)) {
diff --git a/launcher/minecraft/launch/MinecraftServerTarget.h b/launcher/minecraft/launch/MinecraftServerTarget.h
index af8d6550b..2edd8a30d 100644
--- a/launcher/minecraft/launch/MinecraftServerTarget.h
+++ b/launcher/minecraft/launch/MinecraftServerTarget.h
@@ -26,4 +26,4 @@ struct MinecraftServerTarget {
static MinecraftServerTarget parse(const QString& fullAddress);
};
-typedef std::shared_ptr MinecraftServerTargetPtr;
+using MinecraftServerTargetPtr = std::shared_ptr;
diff --git a/launcher/minecraft/mod/DataPack.cpp b/launcher/minecraft/mod/DataPack.cpp
index 7bf5a3112..fc2d3f68b 100644
--- a/launcher/minecraft/mod/DataPack.cpp
+++ b/launcher/minecraft/mod/DataPack.cpp
@@ -28,7 +28,7 @@
#include "Version.h"
// Values taken from:
-// https://minecraft.fandom.com/wiki/Tutorials/Creating_a_data_pack#%22pack_format%22
+// https://minecraft.wiki/w/Tutorials/Creating_a_data_pack#%22pack_format%22
static const QMap> s_pack_format_versions = {
{ 4, { Version("1.13"), Version("1.14.4") } }, { 5, { Version("1.15"), Version("1.16.1") } },
{ 6, { Version("1.16.2"), Version("1.16.5") } }, { 7, { Version("1.17"), Version("1.17.1") } },
diff --git a/launcher/minecraft/mod/DataPack.h b/launcher/minecraft/mod/DataPack.h
index fc2703c7a..b3787b238 100644
--- a/launcher/minecraft/mod/DataPack.h
+++ b/launcher/minecraft/mod/DataPack.h
@@ -63,7 +63,7 @@ class DataPack : public Resource {
mutable QMutex m_data_lock;
/* The 'version' of a data pack, as defined in the pack.mcmeta file.
- * See https://minecraft.fandom.com/wiki/Data_pack#pack.mcmeta
+ * See https://minecraft.wiki/w/Data_pack#pack.mcmeta
*/
int m_pack_format = 0;
diff --git a/launcher/minecraft/mod/MetadataHandler.h b/launcher/minecraft/mod/MetadataHandler.h
index 88e9ff2b6..3496da2a0 100644
--- a/launcher/minecraft/mod/MetadataHandler.h
+++ b/launcher/minecraft/mod/MetadataHandler.h
@@ -31,6 +31,7 @@ class Mod;
class Metadata {
public:
using ModStruct = Packwiz::V1::Mod;
+ using ModSide = Packwiz::V1::Side;
static auto create(QDir& index_dir, ModPlatform::IndexedPack& mod_pack, ModPlatform::IndexedVersion& mod_version) -> ModStruct
{
diff --git a/launcher/minecraft/mod/Mod.cpp b/launcher/minecraft/mod/Mod.cpp
index ae3dea8d8..310946379 100644
--- a/launcher/minecraft/mod/Mod.cpp
+++ b/launcher/minecraft/mod/Mod.cpp
@@ -132,17 +132,23 @@ auto Mod::destroy(QDir& index_dir, bool preserve_metadata, bool attempt_trash) -
if (!preserve_metadata) {
qDebug() << QString("Destroying metadata for '%1' on purpose").arg(name());
- if (metadata()) {
- Metadata::remove(index_dir, metadata()->slug);
- } else {
- auto n = name();
- Metadata::remove(index_dir, n);
- }
+ destroyMetadata(index_dir);
}
return Resource::destroy(attempt_trash);
}
+void Mod::destroyMetadata(QDir& index_dir)
+{
+ if (metadata()) {
+ Metadata::remove(index_dir, metadata()->slug);
+ } else {
+ auto n = name();
+ Metadata::remove(index_dir, n);
+ }
+ m_local_details.metadata = nullptr;
+}
+
auto Mod::details() const -> const ModDetails&
{
return m_local_details;
@@ -246,7 +252,8 @@ void Mod::setIcon(QImage new_image) const
PixmapCache::remove(m_pack_image_cache_key.key);
// scale the image to avoid flooding the pixmapcache
- auto pixmap = QPixmap::fromImage(new_image.scaled({ 64, 64 }, Qt::AspectRatioMode::KeepAspectRatioByExpanding));
+ auto pixmap =
+ QPixmap::fromImage(new_image.scaled({ 64, 64 }, Qt::AspectRatioMode::KeepAspectRatioByExpanding, Qt::SmoothTransformation));
m_pack_image_cache_key.key = PixmapCache::insert(pixmap);
m_pack_image_cache_key.was_ever_used = true;
@@ -259,7 +266,7 @@ QPixmap Mod::icon(QSize size, Qt::AspectRatioMode mode) const
if (PixmapCache::find(m_pack_image_cache_key.key, &cached_image)) {
if (size.isNull())
return cached_image;
- return cached_image.scaled(size, mode);
+ return cached_image.scaled(size, mode, Qt::SmoothTransformation);
}
// No valid image we can get
diff --git a/launcher/minecraft/mod/Mod.h b/launcher/minecraft/mod/Mod.h
index 6dafecfc5..e97ee9d3b 100644
--- a/launcher/minecraft/mod/Mod.h
+++ b/launcher/minecraft/mod/Mod.h
@@ -93,6 +93,8 @@ class Mod : public Resource {
// Delete all the files of this mod
auto destroy(QDir& index_dir, bool preserve_metadata = false, bool attempt_trash = true) -> bool;
+ // Delete the metadata only
+ void destroyMetadata(QDir& index_dir);
void finishResolvingWithDetails(ModDetails&& details);
diff --git a/launcher/minecraft/mod/ModFolderModel.cpp b/launcher/minecraft/mod/ModFolderModel.cpp
index eed35615c..11f7cd0f1 100644
--- a/launcher/minecraft/mod/ModFolderModel.cpp
+++ b/launcher/minecraft/mod/ModFolderModel.cpp
@@ -131,6 +131,11 @@ QVariant ModFolderModel::data(const QModelIndex& index, int role) const
}
return {};
}
+ case Qt::SizeHintRole:
+ if (column == ImageColumn) {
+ return QSize(32, 32);
+ }
+ return {};
case Qt::CheckStateRole:
switch (column) {
case ActiveColumn:
@@ -233,6 +238,25 @@ bool ModFolderModel::deleteMods(const QModelIndexList& indexes)
return true;
}
+bool ModFolderModel::deleteModsMetadata(const QModelIndexList& indexes)
+{
+ if (indexes.isEmpty())
+ return true;
+
+ for (auto i : indexes) {
+ if (i.column() != 0) {
+ continue;
+ }
+ auto m = at(i.row());
+ auto index_dir = indexDir();
+ m->destroyMetadata(index_dir);
+ }
+
+ update();
+
+ return true;
+}
+
bool ModFolderModel::isValid()
{
return m_dir.exists() && m_dir.isReadable();
diff --git a/launcher/minecraft/mod/ModFolderModel.h b/launcher/minecraft/mod/ModFolderModel.h
index f1890e87e..61d840f9b 100644
--- a/launcher/minecraft/mod/ModFolderModel.h
+++ b/launcher/minecraft/mod/ModFolderModel.h
@@ -81,6 +81,7 @@ class ModFolderModel : public ResourceFolderModel {
/// Deletes all the selected mods
bool deleteMods(const QModelIndexList& indexes);
+ bool deleteModsMetadata(const QModelIndexList& indexes);
bool isValid();
diff --git a/launcher/minecraft/mod/ResourceFolderModel.cpp b/launcher/minecraft/mod/ResourceFolderModel.cpp
index d3237b34b..0503b660b 100644
--- a/launcher/minecraft/mod/ResourceFolderModel.cpp
+++ b/launcher/minecraft/mod/ResourceFolderModel.cpp
@@ -33,6 +33,10 @@ ResourceFolderModel::ResourceFolderModel(QDir dir, BaseInstance* instance, QObje
connect(&m_watcher, &QFileSystemWatcher::directoryChanged, this, &ResourceFolderModel::directoryChanged);
connect(&m_helper_thread_task, &ConcurrentTask::finished, this, [this] { m_helper_thread_task.clear(); });
+#ifndef LAUNCHER_TEST
+ // in tests the application macro doesn't work
+ m_helper_thread_task.setMaxConcurrent(APPLICATION->settings()->get("NumberOfConcurrentTasks").toInt());
+#endif
}
ResourceFolderModel::~ResourceFolderModel()
diff --git a/launcher/minecraft/mod/ResourceFolderModel.h b/launcher/minecraft/mod/ResourceFolderModel.h
index 80c31e456..60b8879c0 100644
--- a/launcher/minecraft/mod/ResourceFolderModel.h
+++ b/launcher/minecraft/mod/ResourceFolderModel.h
@@ -330,7 +330,8 @@ void ResourceFolderModel::applyUpdates(QSet& current_set, QSet
// When you have a Qt build with assertions turned on, proceeding here will abort the application
if (added_set.size() > 0) {
- beginInsertRows(QModelIndex(), m_resources.size(), m_resources.size() + added_set.size() - 1);
+ beginInsertRows(QModelIndex(), static_cast(m_resources.size()),
+ static_cast(m_resources.size() + added_set.size() - 1));
for (auto& added : added_set) {
auto res = new_resources[added];
diff --git a/launcher/minecraft/mod/ResourcePack.cpp b/launcher/minecraft/mod/ResourcePack.cpp
index dab0f6d67..074534405 100644
--- a/launcher/minecraft/mod/ResourcePack.cpp
+++ b/launcher/minecraft/mod/ResourcePack.cpp
@@ -11,7 +11,7 @@
#include "minecraft/mod/tasks/LocalResourcePackParseTask.h"
// Values taken from:
-// https://minecraft.fandom.com/wiki/Tutorials/Creating_a_resource_pack#Formatting_pack.mcmeta
+// https://minecraft.wiki/w/Tutorials/Creating_a_resource_pack#Formatting_pack.mcmeta
static const QMap> s_pack_format_versions = {
{ 1, { Version("1.6.1"), Version("1.8.9") } }, { 2, { Version("1.9"), Version("1.10.2") } },
{ 3, { Version("1.11"), Version("1.12.2") } }, { 4, { Version("1.13"), Version("1.14.4") } },
@@ -50,7 +50,8 @@ void ResourcePack::setImage(QImage new_image) const
PixmapCache::instance().remove(m_pack_image_cache_key.key);
// scale the image to avoid flooding the pixmapcache
- auto pixmap = QPixmap::fromImage(new_image.scaled({ 64, 64 }, Qt::AspectRatioMode::KeepAspectRatioByExpanding));
+ auto pixmap =
+ QPixmap::fromImage(new_image.scaled({ 64, 64 }, Qt::AspectRatioMode::KeepAspectRatioByExpanding, Qt::SmoothTransformation));
m_pack_image_cache_key.key = PixmapCache::instance().insert(pixmap);
m_pack_image_cache_key.was_ever_used = true;
@@ -68,7 +69,7 @@ QPixmap ResourcePack::image(QSize size, Qt::AspectRatioMode mode) const
if (PixmapCache::instance().find(m_pack_image_cache_key.key, &cached_image)) {
if (size.isNull())
return cached_image;
- return cached_image.scaled(size, mode);
+ return cached_image.scaled(size, mode, Qt::SmoothTransformation);
}
// No valid image we can get
diff --git a/launcher/minecraft/mod/ResourcePack.h b/launcher/minecraft/mod/ResourcePack.h
index da354bc1c..c06f3793d 100644
--- a/launcher/minecraft/mod/ResourcePack.h
+++ b/launcher/minecraft/mod/ResourcePack.h
@@ -51,7 +51,7 @@ class ResourcePack : public Resource {
mutable QMutex m_data_lock;
/* The 'version' of a resource pack, as defined in the pack.mcmeta file.
- * See https://minecraft.fandom.com/wiki/Tutorials/Creating_a_resource_pack#Formatting_pack.mcmeta
+ * See https://minecraft.wiki/w/Tutorials/Creating_a_resource_pack#Formatting_pack.mcmeta
*/
int m_pack_format = 0;
diff --git a/launcher/minecraft/mod/ResourcePackFolderModel.cpp b/launcher/minecraft/mod/ResourcePackFolderModel.cpp
index f27431576..efe1cc5dd 100644
--- a/launcher/minecraft/mod/ResourcePackFolderModel.cpp
+++ b/launcher/minecraft/mod/ResourcePackFolderModel.cpp
@@ -117,6 +117,11 @@ QVariant ResourcePackFolderModel::data(const QModelIndex& index, int role) const
}
return m_resources[row]->internal_id();
}
+ case Qt::SizeHintRole:
+ if (column == ImageColumn) {
+ return QSize(32, 32);
+ }
+ return {};
case Qt::CheckStateRole:
switch (column) {
case ActiveColumn:
diff --git a/launcher/minecraft/mod/ShaderPack.cpp b/launcher/minecraft/mod/ShaderPack.cpp
index 6a9641de2..2c094f26a 100644
--- a/launcher/minecraft/mod/ShaderPack.cpp
+++ b/launcher/minecraft/mod/ShaderPack.cpp
@@ -22,7 +22,7 @@
#include "ShaderPack.h"
-#include "minecraft/mod/tasks/LocalShaderPackParseTask.h"
+#include
void ShaderPack::setPackFormat(ShaderPackFormat new_format)
{
@@ -35,3 +35,8 @@ bool ShaderPack::valid() const
{
return m_pack_format != ShaderPackFormat::INVALID;
}
+
+bool ShaderPack::applyFilter(QRegularExpression filter) const
+{
+ return valid() && Resource::applyFilter(filter);
+}
diff --git a/launcher/minecraft/mod/ShaderPack.h b/launcher/minecraft/mod/ShaderPack.h
index ec0f9404e..d07c124be 100644
--- a/launcher/minecraft/mod/ShaderPack.h
+++ b/launcher/minecraft/mod/ShaderPack.h
@@ -54,6 +54,7 @@ class ShaderPack : public Resource {
void setPackFormat(ShaderPackFormat new_format);
bool valid() const override;
+ [[nodiscard]] bool applyFilter(QRegularExpression filter) const override;
protected:
mutable QMutex m_data_lock;
diff --git a/launcher/minecraft/mod/ShaderPackFolderModel.h b/launcher/minecraft/mod/ShaderPackFolderModel.h
index 44ed37a47..186d02139 100644
--- a/launcher/minecraft/mod/ShaderPackFolderModel.h
+++ b/launcher/minecraft/mod/ShaderPackFolderModel.h
@@ -1,6 +1,9 @@
#pragma once
#include "ResourceFolderModel.h"
+#include "minecraft/mod/ShaderPack.h"
+#include "minecraft/mod/tasks/BasicFolderLoadTask.h"
+#include "minecraft/mod/tasks/LocalShaderPackParseTask.h"
class ShaderPackFolderModel : public ResourceFolderModel {
Q_OBJECT
@@ -9,4 +12,14 @@ class ShaderPackFolderModel : public ResourceFolderModel {
explicit ShaderPackFolderModel(const QString& dir, BaseInstance* instance) : ResourceFolderModel(QDir(dir), instance) {}
virtual QString id() const override { return "shaderpacks"; }
+
+ [[nodiscard]] Task* createUpdateTask() override
+ {
+ return new BasicFolderLoadTask(m_dir, [](QFileInfo const& entry) { return makeShared(entry); });
+ }
+
+ [[nodiscard]] Task* createParseTask(Resource& resource) override
+ {
+ return new LocalShaderPackParseTask(m_next_resolution_ticket, static_cast(resource));
+ }
};
diff --git a/launcher/minecraft/mod/TexturePack.cpp b/launcher/minecraft/mod/TexturePack.cpp
index 7d8c67137..04cc36310 100644
--- a/launcher/minecraft/mod/TexturePack.cpp
+++ b/launcher/minecraft/mod/TexturePack.cpp
@@ -44,7 +44,8 @@ void TexturePack::setImage(QImage new_image) const
PixmapCache::remove(m_pack_image_cache_key.key);
// scale the image to avoid flooding the pixmapcache
- auto pixmap = QPixmap::fromImage(new_image.scaled({ 64, 64 }, Qt::AspectRatioMode::KeepAspectRatioByExpanding));
+ auto pixmap =
+ QPixmap::fromImage(new_image.scaled({ 64, 64 }, Qt::AspectRatioMode::KeepAspectRatioByExpanding, Qt::SmoothTransformation));
m_pack_image_cache_key.key = PixmapCache::insert(pixmap);
m_pack_image_cache_key.was_ever_used = true;
@@ -56,7 +57,7 @@ QPixmap TexturePack::image(QSize size, Qt::AspectRatioMode mode) const
if (PixmapCache::find(m_pack_image_cache_key.key, &cached_image)) {
if (size.isNull())
return cached_image;
- return cached_image.scaled(size, mode);
+ return cached_image.scaled(size, mode, Qt::SmoothTransformation);
}
// No valid image we can get
diff --git a/launcher/minecraft/mod/TexturePackFolderModel.cpp b/launcher/minecraft/mod/TexturePackFolderModel.cpp
index 5c5f2b7c1..e39be1fb9 100644
--- a/launcher/minecraft/mod/TexturePackFolderModel.cpp
+++ b/launcher/minecraft/mod/TexturePackFolderModel.cpp
@@ -104,6 +104,11 @@ QVariant TexturePackFolderModel::data(const QModelIndex& index, int role) const
}
return {};
}
+ case Qt::SizeHintRole:
+ if (column == ImageColumn) {
+ return QSize(32, 32);
+ }
+ return {};
case Qt::CheckStateRole:
if (column == ActiveColumn) {
return m_resources[row]->enabled() ? Qt::Checked : Qt::Unchecked;
diff --git a/launcher/minecraft/mod/tasks/GetModDependenciesTask.cpp b/launcher/minecraft/mod/tasks/GetModDependenciesTask.cpp
index ee84b1f5e..df8c690af 100644
--- a/launcher/minecraft/mod/tasks/GetModDependenciesTask.cpp
+++ b/launcher/minecraft/mod/tasks/GetModDependenciesTask.cpp
@@ -94,7 +94,7 @@ QList GetModDependenciesTask::getDependenciesForVersion
for (auto ver_dep : version.dependencies) {
if (ver_dep.type != ModPlatform::DependencyType::REQUIRED)
continue;
-
+ ver_dep = getOverride(ver_dep, providerName);
auto isOnlyVersion = providerName == ModPlatform::ResourceProvider::MODRINTH && ver_dep.addonId.toString().isEmpty();
if (auto dep = std::find_if(c_dependencies.begin(), c_dependencies.end(),
[&ver_dep, isOnlyVersion](const ModPlatform::Dependency& i) {
@@ -127,7 +127,7 @@ QList GetModDependenciesTask::getDependenciesForVersion
dep != m_pack_dependencies.end()) // check loaded dependencies
continue;
- c_dependencies.append(getOverride(ver_dep, providerName));
+ c_dependencies.append(ver_dep);
}
return c_dependencies;
}
@@ -251,3 +251,32 @@ void GetModDependenciesTask::removePack(const QVariant addonId)
++it;
#endif
}
+
+QHash GetModDependenciesTask::getRequiredBy()
+{
+ QHash rby;
+ auto fullList = m_selected + m_pack_dependencies;
+ for (auto& mod : fullList) {
+ auto addonId = mod->pack->addonId;
+ auto provider = mod->pack->provider;
+ auto version = mod->version.fileId;
+ auto req = QStringList();
+ for (auto& smod : fullList) {
+ if (provider != smod->pack->provider)
+ continue;
+ auto deps = smod->version.dependencies;
+ if (auto dep = std::find_if(deps.begin(), deps.end(),
+ [addonId, provider, version](const ModPlatform::Dependency& d) {
+ return d.type == ModPlatform::DependencyType::REQUIRED &&
+ (provider == ModPlatform::ResourceProvider::MODRINTH && d.addonId.toString().isEmpty()
+ ? version == d.version
+ : d.addonId == addonId);
+ });
+ dep != deps.end()) {
+ req.append(smod->pack->name);
+ }
+ }
+ rby[addonId.toString()] = req;
+ }
+ return rby;
+}
\ No newline at end of file
diff --git a/launcher/minecraft/mod/tasks/GetModDependenciesTask.h b/launcher/minecraft/mod/tasks/GetModDependenciesTask.h
index a8b9953d3..2580f8077 100644
--- a/launcher/minecraft/mod/tasks/GetModDependenciesTask.h
+++ b/launcher/minecraft/mod/tasks/GetModDependenciesTask.h
@@ -62,6 +62,7 @@ class GetModDependenciesTask : public SequentialTask {
QList> selected);
auto getDependecies() const -> QList> { return m_pack_dependencies; }
+ QHash getRequiredBy();
protected slots:
Task::Ptr prepareDependencyTask(const ModPlatform::Dependency&, const ModPlatform::ResourceProvider, int);
diff --git a/launcher/minecraft/mod/tasks/LocalDataPackParseTask.cpp b/launcher/minecraft/mod/tasks/LocalDataPackParseTask.cpp
index 5bb448778..82f6b9df9 100644
--- a/launcher/minecraft/mod/tasks/LocalDataPackParseTask.cpp
+++ b/launcher/minecraft/mod/tasks/LocalDataPackParseTask.cpp
@@ -133,7 +133,7 @@ bool processZIP(DataPack& pack, ProcessingLevel level)
return true;
}
-// https://minecraft.fandom.com/wiki/Data_pack#pack.mcmeta
+// https://minecraft.wiki/w/Data_pack#pack.mcmeta
bool processMCMeta(DataPack& pack, QByteArray&& raw_data)
{
try {
diff --git a/launcher/minecraft/mod/tasks/LocalModParseTask.cpp b/launcher/minecraft/mod/tasks/LocalModParseTask.cpp
index 7ec00c0c6..e9e12d86a 100644
--- a/launcher/minecraft/mod/tasks/LocalModParseTask.cpp
+++ b/launcher/minecraft/mod/tasks/LocalModParseTask.cpp
@@ -688,6 +688,7 @@ bool loadIconFile(const Mod& mod)
return png_invalid(); // icon invalid
}
}
+ return false;
}
case ResourceType::ZIPFILE: {
QuaZip zip(mod.fileinfo().filePath());
@@ -714,6 +715,7 @@ bool loadIconFile(const Mod& mod)
} else {
return png_invalid(); // could not set icon as current file.
}
+ return false;
}
case ResourceType::LITEMOD: {
return false; // can lightmods even have icons?
diff --git a/launcher/minecraft/mod/tasks/LocalResourcePackParseTask.cpp b/launcher/minecraft/mod/tasks/LocalResourcePackParseTask.cpp
index 73cbf891c..26bc07637 100644
--- a/launcher/minecraft/mod/tasks/LocalResourcePackParseTask.cpp
+++ b/launcher/minecraft/mod/tasks/LocalResourcePackParseTask.cpp
@@ -178,7 +178,7 @@ bool processZIP(ResourcePack& pack, ProcessingLevel level)
return true;
}
-// https://minecraft.fandom.com/wiki/Tutorials/Creating_a_resource_pack#Formatting_pack.mcmeta
+// https://minecraft.wiki/w/Tutorials/Creating_a_resource_pack#Formatting_pack.mcmeta
bool processMCMeta(ResourcePack& pack, QByteArray&& raw_data)
{
try {
@@ -232,10 +232,9 @@ bool processPackPNG(const ResourcePack& pack)
} else {
return png_invalid(); // pack.png does not exists or is not a valid file.
}
+ return false; // not processed correctly; https://github.com/PrismLauncher/PrismLauncher/issues/1740
}
case ResourceType::ZIPFILE: {
- Q_ASSERT(pack.type() == ResourceType::ZIPFILE);
-
QuaZip zip(pack.fileinfo().filePath());
if (!zip.open(QuaZip::mdUnzip))
return false; // can't open zip file
@@ -259,6 +258,7 @@ bool processPackPNG(const ResourcePack& pack)
} else {
return png_invalid(); // could not set pack.mcmeta as current file.
}
+ return false; // not processed correctly; https://github.com/PrismLauncher/PrismLauncher/issues/1740
}
default:
qWarning() << "Invalid type for resource pack parse task!";
diff --git a/launcher/minecraft/mod/tasks/LocalTexturePackParseTask.cpp b/launcher/minecraft/mod/tasks/LocalTexturePackParseTask.cpp
index 887a1062e..d7e61ca90 100644
--- a/launcher/minecraft/mod/tasks/LocalTexturePackParseTask.cpp
+++ b/launcher/minecraft/mod/tasks/LocalTexturePackParseTask.cpp
@@ -186,10 +186,9 @@ bool processPackPNG(const TexturePack& pack)
} else {
return png_invalid(); // pack.png does not exists or is not a valid file.
}
+ return false;
}
case ResourceType::ZIPFILE: {
- Q_ASSERT(pack.type() == ResourceType::ZIPFILE);
-
QuaZip zip(pack.fileinfo().filePath());
if (!zip.open(QuaZip::mdUnzip))
return false; // can't open zip file
@@ -215,6 +214,7 @@ bool processPackPNG(const TexturePack& pack)
zip.close();
return png_invalid(); // could not set pack.mcmeta as current file.
}
+ return false;
}
default:
qWarning() << "Invalid type for resource pack parse task!";
diff --git a/launcher/minecraft/mod/tasks/ModFolderLoadTask.cpp b/launcher/minecraft/mod/tasks/ModFolderLoadTask.cpp
index 9f79ba098..2094df4fc 100644
--- a/launcher/minecraft/mod/tasks/ModFolderLoadTask.cpp
+++ b/launcher/minecraft/mod/tasks/ModFolderLoadTask.cpp
@@ -122,7 +122,7 @@ void ModFolderLoadTask::getFromMetadata()
auto metadata = Metadata::get(m_index_dir, entry);
if (!metadata.isValid()) {
- return;
+ continue;
}
auto* mod = new Mod(m_mods_dir, metadata);
diff --git a/launcher/minecraft/services/SkinDelete.h b/launcher/minecraft/services/SkinDelete.h
index 853de12ef..40664a645 100644
--- a/launcher/minecraft/services/SkinDelete.h
+++ b/launcher/minecraft/services/SkinDelete.h
@@ -5,7 +5,7 @@
#include
#include "tasks/Task.h"
-typedef shared_qobject_ptr SkinDeletePtr;
+using SkinDeletePtr = shared_qobject_ptr;
class SkinDelete : public Task {
Q_OBJECT
diff --git a/launcher/minecraft/services/SkinUpload.h b/launcher/minecraft/services/SkinUpload.h
index c986d2a7d..75f1551bb 100644
--- a/launcher/minecraft/services/SkinUpload.h
+++ b/launcher/minecraft/services/SkinUpload.h
@@ -1,12 +1,12 @@
#pragma once
+#include
#include
#include
#include
#include "tasks/Task.h"
-#include
-typedef shared_qobject_ptr SkinUploadPtr;
+using SkinUploadPtr = shared_qobject_ptr;
class SkinUpload : public Task {
Q_OBJECT
diff --git a/launcher/modplatform/CheckUpdateTask.h b/launcher/modplatform/CheckUpdateTask.h
index d125a5879..8bd83d988 100644
--- a/launcher/modplatform/CheckUpdateTask.h
+++ b/launcher/modplatform/CheckUpdateTask.h
@@ -1,6 +1,7 @@
#pragma once
#include "minecraft/mod/Mod.h"
+#include "minecraft/mod/tasks/GetModDependenciesTask.h"
#include "modplatform/ModIndex.h"
#include "modplatform/ResourceAPI.h"
#include "tasks/Task.h"
@@ -23,6 +24,7 @@ class CheckUpdateTask : public Task {
QString old_hash;
QString old_version;
QString new_version;
+ std::optional new_version_type;
QString changelog;
ModPlatform::ResourceProvider provider;
shared_qobject_ptr download;
@@ -32,14 +34,23 @@ class CheckUpdateTask : public Task {
QString old_h,
QString old_v,
QString new_v,
+ std::optional new_v_type,
QString changelog,
ModPlatform::ResourceProvider p,
shared_qobject_ptr t)
- : name(name), old_hash(old_h), old_version(old_v), new_version(new_v), changelog(changelog), provider(p), download(t)
+ : name(name)
+ , old_hash(old_h)
+ , old_version(old_v)
+ , new_version(new_v)
+ , new_version_type(new_v_type)
+ , changelog(changelog)
+ , provider(p)
+ , download(t)
{}
};
auto getUpdatable() -> std::vector&& { return std::move(m_updatable); }
+ auto getDependencies() -> QList>&& { return std::move(m_deps); }
public slots:
bool abort() override = 0;
@@ -57,4 +68,5 @@ class CheckUpdateTask : public Task {
std::shared_ptr m_mods_folder;
std::vector m_updatable;
+ QList> m_deps;
};
diff --git a/launcher/modplatform/EnsureMetadataTask.cpp b/launcher/modplatform/EnsureMetadataTask.cpp
index c3eadd06d..a9ad22581 100644
--- a/launcher/modplatform/EnsureMetadataTask.cpp
+++ b/launcher/modplatform/EnsureMetadataTask.cpp
@@ -3,6 +3,7 @@
#include
#include
+#include "Application.h"
#include "Json.h"
#include "minecraft/mod/Mod.h"
@@ -33,7 +34,7 @@ EnsureMetadataTask::EnsureMetadataTask(Mod* mod, QDir dir, ModPlatform::Resource
EnsureMetadataTask::EnsureMetadataTask(QList& mods, QDir dir, ModPlatform::ResourceProvider prov)
: Task(nullptr), m_index_dir(dir), m_provider(prov), m_current_task(nullptr)
{
- m_hashing_task.reset(new ConcurrentTask(this, "MakeHashesTask", 10));
+ m_hashing_task.reset(new ConcurrentTask(this, "MakeHashesTask", APPLICATION->settings()->get("NumberOfConcurrentTasks").toInt()));
for (auto* mod : mods) {
auto hash_task = createNewHash(mod);
if (!hash_task)
diff --git a/launcher/modplatform/ModIndex.cpp b/launcher/modplatform/ModIndex.cpp
index e8e4a38dd..fc79dff15 100644
--- a/launcher/modplatform/ModIndex.cpp
+++ b/launcher/modplatform/ModIndex.cpp
@@ -24,6 +24,40 @@
namespace ModPlatform {
+static const QMap s_indexed_version_type_names = {
+ { "release", IndexedVersionType::VersionType::Release },
+ { "beta", IndexedVersionType::VersionType::Beta },
+ { "alpha", IndexedVersionType::VersionType::Alpha }
+};
+
+IndexedVersionType::IndexedVersionType(const QString& type) : IndexedVersionType(enumFromString(type)) {}
+
+IndexedVersionType::IndexedVersionType(const IndexedVersionType::VersionType& type)
+{
+ m_type = type;
+}
+
+IndexedVersionType::IndexedVersionType(const IndexedVersionType& other)
+{
+ m_type = other.m_type;
+}
+
+IndexedVersionType& IndexedVersionType::operator=(const IndexedVersionType& other)
+{
+ m_type = other.m_type;
+ return *this;
+}
+
+const QString IndexedVersionType::toString(const IndexedVersionType::VersionType& type)
+{
+ return s_indexed_version_type_names.key(type, "unknown");
+}
+
+IndexedVersionType::VersionType IndexedVersionType::enumFromString(const QString& type)
+{
+ return s_indexed_version_type_names.value(type, IndexedVersionType::VersionType::Unknown);
+}
+
auto ProviderCapabilities::name(ResourceProvider p) -> const char*
{
switch (p) {
diff --git a/launcher/modplatform/ModIndex.h b/launcher/modplatform/ModIndex.h
index 7d144176d..72294c399 100644
--- a/launcher/modplatform/ModIndex.h
+++ b/launcher/modplatform/ModIndex.h
@@ -25,6 +25,7 @@
#include
#include
#include
+#include
class QIODevice;
@@ -58,6 +59,34 @@ struct DonationData {
QString url;
};
+struct IndexedVersionType {
+ enum class VersionType { Release = 1, Beta, Alpha, Unknown };
+ IndexedVersionType(const QString& type);
+ IndexedVersionType(const IndexedVersionType::VersionType& type);
+ IndexedVersionType(const IndexedVersionType& type);
+ IndexedVersionType() : IndexedVersionType(IndexedVersionType::VersionType::Unknown) {}
+ static const QString toString(const IndexedVersionType::VersionType& type);
+ static IndexedVersionType::VersionType enumFromString(const QString& type);
+ bool isValid() const { return m_type != IndexedVersionType::VersionType::Unknown; }
+ IndexedVersionType& operator=(const IndexedVersionType& other);
+ bool operator==(const IndexedVersionType& other) const { return m_type == other.m_type; }
+ bool operator==(const IndexedVersionType::VersionType& type) const { return m_type == type; }
+ bool operator!=(const IndexedVersionType& other) const { return m_type != other.m_type; }
+ bool operator!=(const IndexedVersionType::VersionType& type) const { return m_type != type; }
+ bool operator<(const IndexedVersionType& other) const { return m_type < other.m_type; }
+ bool operator<(const IndexedVersionType::VersionType& type) const { return m_type < type; }
+ bool operator<=(const IndexedVersionType& other) const { return m_type <= other.m_type; }
+ bool operator<=(const IndexedVersionType::VersionType& type) const { return m_type <= type; }
+ bool operator>(const IndexedVersionType& other) const { return m_type > other.m_type; }
+ bool operator>(const IndexedVersionType::VersionType& type) const { return m_type > type; }
+ bool operator>=(const IndexedVersionType& other) const { return m_type >= other.m_type; }
+ bool operator>=(const IndexedVersionType::VersionType& type) const { return m_type >= type; }
+
+ QString toString() const { return toString(m_type); }
+
+ IndexedVersionType::VersionType m_type;
+};
+
struct Dependency {
QVariant addonId;
DependencyType type;
@@ -69,6 +98,7 @@ struct IndexedVersion {
QVariant fileId;
QString version;
QString version_number = {};
+ IndexedVersionType version_type;
QStringList mcVersion;
QString downloadUrl;
QString date;
@@ -107,6 +137,7 @@ struct IndexedPack {
QString logoName;
QString logoUrl;
QString websiteUrl;
+ QString side;
bool versionsLoaded = false;
QVector versions;
diff --git a/launcher/modplatform/atlauncher/ATLPackIndex.cpp b/launcher/modplatform/atlauncher/ATLPackIndex.cpp
index 3be169739..678db63cc 100644
--- a/launcher/modplatform/atlauncher/ATLPackIndex.cpp
+++ b/launcher/modplatform/atlauncher/ATLPackIndex.cpp
@@ -43,5 +43,5 @@ void ATLauncher::loadIndexedPack(ATLauncher::IndexedPack& m, QJsonObject& obj)
m.system = Json::ensureBoolean(obj, QString("system"), false);
m.description = Json::ensureString(obj, "description", "");
- m.safeName = Json::requireString(obj, "name").replace(QRegularExpression("[^A-Za-z0-9]"), "");
+ m.safeName = Json::requireString(obj, "name").replace(QRegularExpression("[^A-Za-z0-9]"), "").toLower() + ".png";
}
diff --git a/launcher/modplatform/atlauncher/ATLPackInstallTask.cpp b/launcher/modplatform/atlauncher/ATLPackInstallTask.cpp
index e5771b7cd..8ae8145de 100644
--- a/launcher/modplatform/atlauncher/ATLPackInstallTask.cpp
+++ b/launcher/modplatform/atlauncher/ATLPackInstallTask.cpp
@@ -37,6 +37,7 @@
#include "ATLPackInstallTask.h"
#include
+#include
#include
@@ -50,6 +51,7 @@
#include "minecraft/MinecraftInstance.h"
#include "minecraft/OneSixVersionFormat.h"
#include "minecraft/PackProfile.h"
+#include "modplatform/atlauncher/ATLPackManifest.h"
#include "net/ChecksumValidator.h"
#include "settings/INISettingsObject.h"
@@ -57,6 +59,7 @@
#include "Application.h"
#include "BuildConfig.h"
+#include "ui/dialogs/BlockedModsDialog.h"
namespace ATLauncher {
@@ -717,6 +720,8 @@ void PackInstallTask::downloadMods()
jarmods.clear();
jobPtr.reset(new NetJob(tr("Mod download"), APPLICATION->network()));
+
+ QList blocked_mods;
for (const auto& mod : m_version.mods) {
// skip non-client mods
if (!mod.client)
@@ -731,9 +736,10 @@ void PackInstallTask::downloadMods()
case DownloadType::Server:
url = BuildConfig.ATL_DOWNLOAD_SERVER_URL + mod.url;
break;
- case DownloadType::Browser:
- emitFailed(tr("Unsupported download type: %1").arg(mod.download_raw));
- return;
+ case DownloadType::Browser: {
+ blocked_mods.append(mod);
+ continue;
+ }
case DownloadType::Direct:
url = mod.url;
break;
@@ -805,24 +811,86 @@ void PackInstallTask::downloadMods()
modsToCopy[entry->getFullPath()] = path;
}
}
+ if (!blocked_mods.isEmpty()) {
+ QList mods;
+
+ for (auto mod : blocked_mods) {
+ BlockedMod blocked_mod;
+ blocked_mod.name = mod.file;
+ blocked_mod.websiteUrl = mod.url;
+ blocked_mod.hash = mod.md5;
+ blocked_mod.matched = false;
+ blocked_mod.localPath = "";
+
+ mods.append(blocked_mod);
+ }
+
+ qWarning() << "Blocked mods found, displaying mod list";
+
+ BlockedModsDialog message_dialog(nullptr, tr("Blocked mods found"),
+ tr("The following files are not available for download in third party launchers.
"
+ "You will need to manually download them and add them to the instance."),
+ mods, "md5");
+
+ message_dialog.setModal(true);
+
+ if (message_dialog.exec()) {
+ qDebug() << "Post dialog blocked mods list: " << mods;
+ for (auto blocked : mods) {
+ if (!blocked.matched) {
+ qDebug() << blocked.name << "was not matched to a local file, skipping copy";
+ continue;
+ }
+ auto modIter = std::find_if(blocked_mods.begin(), blocked_mods.end(),
+ [blocked](const VersionMod& mod) { return mod.url == blocked.websiteUrl; });
+ if (modIter == blocked_mods.end())
+ continue;
+ auto mod = *modIter;
+ if (mod.type == ModType::Extract || mod.type == ModType::TexturePackExtract || mod.type == ModType::ResourcePackExtract) {
+ modsToExtract.insert(blocked.localPath, mod);
+ } else if (mod.type == ModType::Decomp) {
+ modsToDecomp.insert(blocked.localPath, mod);
+ } else {
+ auto relpath = getDirForModType(mod.type, mod.type_raw);
+ if (relpath == Q_NULLPTR)
+ continue;
+
+ auto path = FS::PathCombine(m_stagingPath, "minecraft", relpath, mod.file);
+
+ if (mod.type == ModType::Forge) {
+ auto ver = getComponentVersion("net.minecraftforge", mod.version);
+ if (ver) {
+ componentsToInstall.insert("net.minecraftforge", ver);
+ continue;
+ }
+
+ qDebug() << "Jarmod: " + path;
+ jarmods.push_back(path);
+ }
+
+ if (mod.type == ModType::Jar) {
+ qDebug() << "Jarmod: " + path;
+ jarmods.push_back(path);
+ }
+
+ modsToCopy[blocked.localPath] = path;
+ }
+ }
+ } else {
+ emitFailed(tr("Unknown download type: %1").arg("browser"));
+ return;
+ }
+ }
connect(jobPtr.get(), &NetJob::succeeded, this, &PackInstallTask::onModsDownloaded);
- connect(jobPtr.get(), &NetJob::failed, [&](QString reason) {
- abortable = false;
- jobPtr.reset();
- emitFailed(reason);
- });
- connect(jobPtr.get(), &NetJob::progress, [&](qint64 current, qint64 total) {
+ connect(jobPtr.get(), &NetJob::progress, [this](qint64 current, qint64 total) {
setDetails(tr("%1 out of %2 complete").arg(current).arg(total));
abortable = true;
setProgress(current, total);
});
connect(jobPtr.get(), &NetJob::stepProgress, this, &PackInstallTask::propagateStepProgress);
- connect(jobPtr.get(), &NetJob::aborted, [&] {
- abortable = false;
- jobPtr.reset();
- emitAborted();
- });
+ connect(jobPtr.get(), &NetJob::aborted, &PackInstallTask::emitAborted);
+ connect(jobPtr.get(), &NetJob::failed, &PackInstallTask::emitFailed);
jobPtr->start();
}
@@ -843,7 +911,7 @@ void PackInstallTask::onModsDownloaded()
QtConcurrent::run(QThreadPool::globalInstance(), this, &PackInstallTask::extractMods, modsToExtract, modsToDecomp, modsToCopy);
#endif
connect(&m_modExtractFutureWatcher, &QFutureWatcher::finished, this, &PackInstallTask::onModsExtracted);
- connect(&m_modExtractFutureWatcher, &QFutureWatcher::canceled, this, [&]() { emitAborted(); });
+ connect(&m_modExtractFutureWatcher, &QFutureWatcher::canceled, this, &PackInstallTask::emitAborted);
m_modExtractFutureWatcher.setFuture(m_modExtractFuture);
} else {
install();
diff --git a/launcher/modplatform/flame/FlameAPI.cpp b/launcher/modplatform/flame/FlameAPI.cpp
index e99ce3a56..bb4f18983 100644
--- a/launcher/modplatform/flame/FlameAPI.cpp
+++ b/launcher/modplatform/flame/FlameAPI.cpp
@@ -208,17 +208,15 @@ Task::Ptr FlameAPI::getFile(const QString& addonId, const QString& fileId, std::
return netJob;
}
-// https://docs.curseforge.com/?python#tocS_ModsSearchSortField
-static QList s_sorts = { { 1, "Featured", QObject::tr("Sort by Featured") },
- { 2, "Popularity", QObject::tr("Sort by Popularity") },
- { 3, "LastUpdated", QObject::tr("Sort by Last Updated") },
- { 4, "Name", QObject::tr("Sort by Name") },
- { 5, "Author", QObject::tr("Sort by Author") },
- { 6, "TotalDownloads", QObject::tr("Sort by Downloads") },
- { 7, "Category", QObject::tr("Sort by Category") },
- { 8, "GameVersion", QObject::tr("Sort by Game Version") } };
-
QList FlameAPI::getSortingMethods() const
{
- return s_sorts;
+ // https://docs.curseforge.com/?python#tocS_ModsSearchSortField
+ return { { 1, "Featured", QObject::tr("Sort by Featured") },
+ { 2, "Popularity", QObject::tr("Sort by Popularity") },
+ { 3, "LastUpdated", QObject::tr("Sort by Last Updated") },
+ { 4, "Name", QObject::tr("Sort by Name") },
+ { 5, "Author", QObject::tr("Sort by Author") },
+ { 6, "TotalDownloads", QObject::tr("Sort by Downloads") },
+ { 7, "Category", QObject::tr("Sort by Category") },
+ { 8, "GameVersion", QObject::tr("Sort by Game Version") } };
}
diff --git a/launcher/modplatform/flame/FlameAPI.h b/launcher/modplatform/flame/FlameAPI.h
index 47350c33e..e22d8f0d8 100644
--- a/launcher/modplatform/flame/FlameAPI.h
+++ b/launcher/modplatform/flame/FlameAPI.h
@@ -38,6 +38,8 @@ class FlameAPI : public NetworkResourceAPI {
return 6;
case ModPlatform::ResourceType::RESOURCE_PACK:
return 12;
+ case ModPlatform::ResourceType::SHADER_PACK:
+ return 6552;
}
}
diff --git a/launcher/modplatform/flame/FlameCheckUpdate.cpp b/launcher/modplatform/flame/FlameCheckUpdate.cpp
index 476a4667a..c014863a3 100644
--- a/launcher/modplatform/flame/FlameCheckUpdate.cpp
+++ b/launcher/modplatform/flame/FlameCheckUpdate.cpp
@@ -10,6 +10,7 @@
#include "ResourceDownloadTask.h"
#include "minecraft/mod/ModFolderModel.h"
+#include "minecraft/mod/tasks/GetModDependenciesTask.h"
#include "net/ApiDownload.h"
@@ -154,18 +155,17 @@ void FlameCheckUpdate::executeTask()
continue;
}
+ // Fake pack with the necessary info to pass to the download task :)
+ auto pack = std::make_shared();
+ pack->name = mod->name();
+ pack->slug = mod->metadata()->slug;
+ pack->addonId = mod->metadata()->project_id;
+ pack->websiteUrl = mod->homeurl();
+ for (auto& author : mod->authors())
+ pack->authors.append({ author });
+ pack->description = mod->description();
+ pack->provider = ModPlatform::ResourceProvider::FLAME;
if (!latest_ver.hash.isEmpty() && (mod->metadata()->hash != latest_ver.hash || mod->status() == ModStatus::NotInstalled)) {
- // Fake pack with the necessary info to pass to the download task :)
- auto pack = std::make_shared();
- pack->name = mod->name();
- pack->slug = mod->metadata()->slug;
- pack->addonId = mod->metadata()->project_id;
- pack->websiteUrl = mod->homeurl();
- for (auto& author : mod->authors())
- pack->authors.append({ author });
- pack->description = mod->description();
- pack->provider = ModPlatform::ResourceProvider::FLAME;
-
auto old_version = mod->version();
if (old_version.isEmpty() && mod->status() != ModStatus::NotInstalled) {
auto current_ver = getFileInfo(latest_ver.addonId.toInt(), mod->metadata()->file_id.toInt());
@@ -173,10 +173,11 @@ void FlameCheckUpdate::executeTask()
}
auto download_task = makeShared(pack, latest_ver, m_mods_folder);
- m_updatable.emplace_back(pack->name, mod->metadata()->hash, old_version, latest_ver.version,
+ m_updatable.emplace_back(pack->name, mod->metadata()->hash, old_version, latest_ver.version, latest_ver.version_type,
api.getModFileChangelog(latest_ver.addonId.toInt(), latest_ver.fileId.toInt()),
ModPlatform::ResourceProvider::FLAME, download_task);
}
+ m_deps.append(std::make_shared(pack, latest_ver));
}
emitSucceeded();
diff --git a/launcher/modplatform/flame/FlameInstanceCreationTask.cpp b/launcher/modplatform/flame/FlameInstanceCreationTask.cpp
index 45b4e2125..2a26ce944 100644
--- a/launcher/modplatform/flame/FlameInstanceCreationTask.cpp
+++ b/launcher/modplatform/flame/FlameInstanceCreationTask.cpp
@@ -62,6 +62,7 @@
#include "minecraft/World.h"
#include "minecraft/mod/tasks/LocalResourceParse.h"
#include "net/ApiDownload.h"
+#include "ui/pages/modplatform/OptionalModDialog.h"
static const FlameAPI api;
@@ -509,13 +510,33 @@ void FlameCreationTask::idResolverSucceeded(QEventLoop& loop)
void FlameCreationTask::setupDownloadJob(QEventLoop& loop)
{
m_files_job.reset(new NetJob(tr("Mod Download Flame"), APPLICATION->network()));
- for (const auto& result : m_mod_id_resolver->getResults().files) {
- QString filename = result.fileName;
+ auto results = m_mod_id_resolver->getResults().files;
+
+ QStringList optionalFiles;
+ for (auto& result : results) {
if (!result.required) {
- filename += ".disabled";
+ optionalFiles << FS::PathCombine(result.targetFolder, result.fileName);
+ }
+ }
+
+ QStringList selectedOptionalMods;
+ if (!optionalFiles.empty()) {
+ OptionalModDialog optionalModDialog(m_parent, optionalFiles);
+ if (optionalModDialog.exec() == QDialog::Rejected) {
+ emitAborted();
+ loop.quit();
+ return;
+ }
+
+ selectedOptionalMods = optionalModDialog.getResult();
+ }
+ for (const auto& result : results) {
+ auto relpath = FS::PathCombine(result.targetFolder, result.fileName);
+ if (!result.required && !selectedOptionalMods.contains(relpath)) {
+ relpath += ".disabled";
}
- auto relpath = FS::PathCombine("minecraft", result.targetFolder, filename);
+ relpath = FS::PathCombine("minecraft", relpath);
auto path = FS::PathCombine(m_stagingPath, relpath);
switch (result.type) {
@@ -547,7 +568,7 @@ void FlameCreationTask::setupDownloadJob(QEventLoop& loop)
m_mod_id_resolver.reset();
connect(m_files_job.get(), &NetJob::succeeded, this, [&]() {
m_files_job.reset();
- validateZIPResouces();
+ validateZIPResources();
});
connect(m_files_job.get(), &NetJob::failed, [&](QString reason) {
m_files_job.reset();
@@ -596,7 +617,7 @@ void FlameCreationTask::copyBlockedMods(QList const& blocked_mods)
setAbortable(true);
}
-void FlameCreationTask::validateZIPResouces()
+void FlameCreationTask::validateZIPResources()
{
qDebug() << "Validating whether resources stored as .zip are in the right place";
for (auto [fileName, targetFolder] : m_ZIP_resources) {
@@ -649,8 +670,8 @@ void FlameCreationTask::validateZIPResouces()
validatePath(fileName, targetFolder, "datapacks");
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
+ // in theory flame API can't do this but who knows, that *may* change ?
+ // better to handle it if it *does* occur in the future
validatePath(fileName, targetFolder, "shaderpacks");
break;
case PackedResourceType::WorldSave:
diff --git a/launcher/modplatform/flame/FlameInstanceCreationTask.h b/launcher/modplatform/flame/FlameInstanceCreationTask.h
index 603d3693e..02ad48f2e 100644
--- a/launcher/modplatform/flame/FlameInstanceCreationTask.h
+++ b/launcher/modplatform/flame/FlameInstanceCreationTask.h
@@ -74,7 +74,7 @@ class FlameCreationTask final : public InstanceCreationTask {
void idResolverSucceeded(QEventLoop&);
void setupDownloadJob(QEventLoop&);
void copyBlockedMods(QList const& blocked_mods);
- void validateZIPResouces();
+ void validateZIPResources();
QString getVersionForLoader(QString uid, QString loaderType, QString version, QString mcVersion);
private:
diff --git a/launcher/modplatform/flame/FlameModIndex.cpp b/launcher/modplatform/flame/FlameModIndex.cpp
index 494dc2a7e..345883c17 100644
--- a/launcher/modplatform/flame/FlameModIndex.cpp
+++ b/launcher/modplatform/flame/FlameModIndex.cpp
@@ -139,6 +139,22 @@ auto FlameMod::loadIndexedPackVersion(QJsonObject& obj, bool load_changelog) ->
file.downloadUrl = Json::ensureString(obj, "downloadUrl");
file.fileName = Json::requireString(obj, "fileName");
+ ModPlatform::IndexedVersionType::VersionType ver_type;
+ switch (Json::requireInteger(obj, "releaseType")) {
+ case 1:
+ ver_type = ModPlatform::IndexedVersionType::VersionType::Release;
+ break;
+ case 2:
+ ver_type = ModPlatform::IndexedVersionType::VersionType::Beta;
+ break;
+ case 3:
+ ver_type = ModPlatform::IndexedVersionType::VersionType::Alpha;
+ break;
+ default:
+ ver_type = ModPlatform::IndexedVersionType::VersionType::Unknown;
+ }
+ file.version_type = ModPlatform::IndexedVersionType(ver_type);
+
auto hash_list = Json::ensureArray(obj, "hashes");
for (auto h : hash_list) {
auto hash_entry = Json::ensureObject(h);
diff --git a/launcher/modplatform/flame/FlamePackExportTask.cpp b/launcher/modplatform/flame/FlamePackExportTask.cpp
index d86d34bf4..b5ab7bc73 100644
--- a/launcher/modplatform/flame/FlamePackExportTask.cpp
+++ b/launcher/modplatform/flame/FlamePackExportTask.cpp
@@ -28,6 +28,7 @@
#include
#include
#include
+#include "Application.h"
#include "Json.h"
#include "MMCZip.h"
#include "minecraft/PackProfile.h"
@@ -102,7 +103,8 @@ void FlamePackExportTask::collectHashes()
setStatus(tr("Finding file hashes..."));
setProgress(1, 5);
auto allMods = mcInstance->loaderModList()->allMods();
- ConcurrentTask::Ptr hashingTask(new ConcurrentTask(this, "MakeHashesTask", 10));
+ ConcurrentTask::Ptr hashingTask(
+ new ConcurrentTask(this, "MakeHashesTask", APPLICATION->settings()->get("NumberOfConcurrentTasks").toInt()));
task.reset(hashingTask);
for (const QFileInfo& file : files) {
const QString relative = gameRoot.relativeFilePath(file.absoluteFilePath());
diff --git a/launcher/modplatform/flame/FlamePackIndex.cpp b/launcher/modplatform/flame/FlamePackIndex.cpp
index 21835a543..ca8e0a853 100644
--- a/launcher/modplatform/flame/FlamePackIndex.cpp
+++ b/launcher/modplatform/flame/FlamePackIndex.cpp
@@ -1,4 +1,6 @@
#include "FlamePackIndex.h"
+#include
+#include
#include "Json.h"
@@ -9,8 +11,8 @@ void Flame::loadIndexedPack(Flame::IndexedPack& pack, QJsonObject& obj)
pack.description = Json::ensureString(obj, "summary", "");
auto logo = Json::requireObject(obj, "logo");
- pack.logoName = Json::requireString(logo, "title");
pack.logoUrl = Json::requireString(logo, "thumbnailUrl");
+ pack.logoName = Json::requireString(obj, "slug") + "." + QFileInfo(QUrl(pack.logoUrl).fileName()).suffix();
auto authors = Json::requireArray(obj, "authors");
for (auto authorIter : authors) {
@@ -89,6 +91,22 @@ void Flame::loadIndexedPackVersions(Flame::IndexedPack& pack, QJsonArray& arr)
// pick the latest version supported
file.mcVersion = versionArray[0].toString();
file.version = Json::requireString(version, "displayName");
+
+ ModPlatform::IndexedVersionType::VersionType ver_type;
+ switch (Json::requireInteger(version, "releaseType")) {
+ case 1:
+ ver_type = ModPlatform::IndexedVersionType::VersionType::Release;
+ break;
+ case 2:
+ ver_type = ModPlatform::IndexedVersionType::VersionType::Beta;
+ break;
+ case 3:
+ ver_type = ModPlatform::IndexedVersionType::VersionType::Alpha;
+ break;
+ default:
+ ver_type = ModPlatform::IndexedVersionType::VersionType::Unknown;
+ }
+ file.version_type = ModPlatform::IndexedVersionType(ver_type);
file.downloadUrl = Json::ensureString(version, "downloadUrl");
// only add if we have a download URL (third party distribution is enabled)
diff --git a/launcher/modplatform/flame/FlamePackIndex.h b/launcher/modplatform/flame/FlamePackIndex.h
index b089b722c..b2a12a67f 100644
--- a/launcher/modplatform/flame/FlamePackIndex.h
+++ b/launcher/modplatform/flame/FlamePackIndex.h
@@ -17,6 +17,7 @@ struct IndexedVersion {
int addonId;
int fileId;
QString version;
+ ModPlatform::IndexedVersionType version_type;
QString mcVersion;
QString downloadUrl;
};
diff --git a/launcher/modplatform/flame/PackManifest.h b/launcher/modplatform/flame/PackManifest.h
index 854cdbc41..4417c2430 100644
--- a/launcher/modplatform/flame/PackManifest.h
+++ b/launcher/modplatform/flame/PackManifest.h
@@ -48,7 +48,7 @@ struct File {
int projectId = 0;
int fileId = 0;
- // NOTE: the opposite to 'optional'. This is at the time of writing unused.
+ // NOTE: the opposite to 'optional'
bool required = true;
QString hash;
// NOTE: only set on blocked files ! Empty otherwise.
diff --git a/launcher/modplatform/import_ftb/PackHelpers.h b/launcher/modplatform/import_ftb/PackHelpers.h
index 5400252b6..221eb5bf6 100644
--- a/launcher/modplatform/import_ftb/PackHelpers.h
+++ b/launcher/modplatform/import_ftb/PackHelpers.h
@@ -45,7 +45,7 @@ struct Modpack {
QIcon icon;
};
-typedef QList ModpackList;
+using ModpackList = QList;
Modpack parseDirectory(QString path);
diff --git a/launcher/modplatform/legacy_ftb/PackHelpers.h b/launcher/modplatform/legacy_ftb/PackHelpers.h
index 4fb535530..f2d18f802 100644
--- a/launcher/modplatform/legacy_ftb/PackHelpers.h
+++ b/launcher/modplatform/legacy_ftb/PackHelpers.h
@@ -31,7 +31,7 @@ struct Modpack {
QString packCode;
};
-typedef QList ModpackList;
+using ModpackList = QList;
} // namespace LegacyFTB
diff --git a/launcher/modplatform/legacy_ftb/PackInstallTask.cpp b/launcher/modplatform/legacy_ftb/PackInstallTask.cpp
index 761f622bb..091296751 100644
--- a/launcher/modplatform/legacy_ftb/PackInstallTask.cpp
+++ b/launcher/modplatform/legacy_ftb/PackInstallTask.cpp
@@ -70,16 +70,18 @@ void PackInstallTask::downloadPack()
setProgress(1, 4);
setAbortable(false);
- archivePath = QString("%1/%2/%3").arg(m_pack.dir, m_version.replace(".", "_"), m_pack.file);
-
+ auto path = QString("%1/%2/%3").arg(m_pack.dir, m_version.replace(".", "_"), m_pack.file);
+ auto entry = APPLICATION->metacache()->resolveEntry("FTBPacks", path);
+ entry->setStale(true);
+ archivePath = entry->getFullPath();
netJobContainer.reset(new NetJob("Download FTB Pack", m_network));
QString url;
if (m_pack.type == PackType::Private) {
- url = QString(BuildConfig.LEGACY_FTB_CDN_BASE_URL + "privatepacks/%1").arg(archivePath);
+ url = QString(BuildConfig.LEGACY_FTB_CDN_BASE_URL + "privatepacks/%1").arg(path);
} else {
- url = QString(BuildConfig.LEGACY_FTB_CDN_BASE_URL + "modpacks/%1").arg(archivePath);
+ url = QString(BuildConfig.LEGACY_FTB_CDN_BASE_URL + "modpacks/%1").arg(path);
}
- netJobContainer->addNetAction(Net::ApiDownload::makeFile(url, archivePath));
+ netJobContainer->addNetAction(Net::ApiDownload::makeCached(url, entry));
connect(netJobContainer.get(), &NetJob::succeeded, this, &PackInstallTask::unzip);
connect(netJobContainer.get(), &NetJob::failed, this, &PackInstallTask::emitFailed);
diff --git a/launcher/modplatform/modrinth/ModrinthAPI.cpp b/launcher/modplatform/modrinth/ModrinthAPI.cpp
index f453f5cb9..8be963ecf 100644
--- a/launcher/modplatform/modrinth/ModrinthAPI.cpp
+++ b/launcher/modplatform/modrinth/ModrinthAPI.cpp
@@ -111,14 +111,12 @@ Task::Ptr ModrinthAPI::getProjects(QStringList addonIds, std::shared_ptr s_sorts = { { 1, "relevance", QObject::tr("Sort by Relevance") },
- { 2, "downloads", QObject::tr("Sort by Downloads") },
- { 3, "follows", QObject::tr("Sort by Follows") },
- { 4, "newest", QObject::tr("Sort by Last Updated") },
- { 5, "updated", QObject::tr("Sort by Newest") } };
-
QList ModrinthAPI::getSortingMethods() const
{
- return s_sorts;
+ // https://docs.modrinth.com/api-spec/#tag/projects/operation/searchProjects
+ return { { 1, "relevance", QObject::tr("Sort by Relevance") },
+ { 2, "downloads", QObject::tr("Sort by Downloads") },
+ { 3, "follows", QObject::tr("Sort by Follows") },
+ { 4, "newest", QObject::tr("Sort by Last Updated") },
+ { 5, "updated", QObject::tr("Sort by Newest") } };
}
diff --git a/launcher/modplatform/modrinth/ModrinthCheckUpdate.cpp b/launcher/modplatform/modrinth/ModrinthCheckUpdate.cpp
index c65f4fa80..9b7c53854 100644
--- a/launcher/modplatform/modrinth/ModrinthCheckUpdate.cpp
+++ b/launcher/modplatform/modrinth/ModrinthCheckUpdate.cpp
@@ -38,7 +38,7 @@ void ModrinthCheckUpdate::executeTask()
QStringList hashes;
auto best_hash_type = ProviderCaps.hashType(ModPlatform::ResourceProvider::MODRINTH).first();
- ConcurrentTask hashing_task(this, "MakeModrinthHashesTask", 10);
+ ConcurrentTask hashing_task(this, "MakeModrinthHashesTask", APPLICATION->settings()->get("NumberOfConcurrentTasks").toInt());
for (auto* mod : m_mods) {
if (!mod->enabled()) {
emit checkFailed(mod, tr("Disabled mods won't be updated, to prevent mod duplication issues!"));
@@ -144,26 +144,27 @@ void ModrinthCheckUpdate::executeTask()
auto mod = *mod_iter;
auto key = project_ver.hash;
+
+ // Fake pack with the necessary info to pass to the download task :)
+ auto pack = std::make_shared();
+ pack->name = mod->name();
+ pack->slug = mod->metadata()->slug;
+ pack->addonId = mod->metadata()->project_id;
+ pack->websiteUrl = mod->homeurl();
+ for (auto& author : mod->authors())
+ pack->authors.append({ author });
+ pack->description = mod->description();
+ pack->provider = ModPlatform::ResourceProvider::MODRINTH;
if ((key != hash && project_ver.is_preferred) || (mod->status() == ModStatus::NotInstalled)) {
if (mod->version() == project_ver.version_number)
continue;
- // Fake pack with the necessary info to pass to the download task :)
- auto pack = std::make_shared();
- pack->name = mod->name();
- pack->slug = mod->metadata()->slug;
- pack->addonId = mod->metadata()->project_id;
- pack->websiteUrl = mod->homeurl();
- for (auto& author : mod->authors())
- pack->authors.append({ author });
- pack->description = mod->description();
- pack->provider = ModPlatform::ResourceProvider::MODRINTH;
-
auto download_task = makeShared(pack, project_ver, m_mods_folder);
- m_updatable.emplace_back(pack->name, hash, mod->version(), project_ver.version_number, project_ver.changelog,
- ModPlatform::ResourceProvider::MODRINTH, download_task);
+ m_updatable.emplace_back(pack->name, hash, mod->version(), project_ver.version_number, project_ver.version_type,
+ project_ver.changelog, ModPlatform::ResourceProvider::MODRINTH, download_task);
}
+ m_deps.append(std::make_shared(pack, project_ver));
}
} catch (Json::JsonException& e) {
failed(e.cause() + " : " + e.what());
diff --git a/launcher/modplatform/modrinth/ModrinthInstanceCreationTask.cpp b/launcher/modplatform/modrinth/ModrinthInstanceCreationTask.cpp
index 9ff6b374d..e732ad39c 100644
--- a/launcher/modplatform/modrinth/ModrinthInstanceCreationTask.cpp
+++ b/launcher/modplatform/modrinth/ModrinthInstanceCreationTask.cpp
@@ -9,6 +9,7 @@
#include "modplatform/helpers/OverrideUtils.h"
+#include "modplatform/modrinth/ModrinthPackManifest.h"
#include "net/ChecksumValidator.h"
#include "net/ApiDownload.h"
@@ -16,8 +17,10 @@
#include "settings/INISettingsObject.h"
#include "ui/dialogs/CustomMessageBox.h"
+#include "ui/pages/modplatform/OptionalModDialog.h"
#include
+#include
bool ModrinthCreationTask::abort()
{
@@ -319,10 +322,10 @@ bool ModrinthCreationTask::parseManifest(const QString& index_path,
}
auto jsonFiles = Json::requireIsArrayOf(obj, "files", "modrinth.index.json");
- bool had_optional = false;
+ std::vector optionalFiles;
for (const auto& modInfo : jsonFiles) {
Modrinth::File file;
- file.path = Json::requireString(modInfo, "path");
+ file.path = Json::requireString(modInfo, "path").replace("\\", "/");
auto env = Json::ensureObject(modInfo, "env");
// 'env' field is optional
@@ -331,18 +334,7 @@ bool ModrinthCreationTask::parseManifest(const QString& index_path,
if (support == "unsupported") {
continue;
} else if (support == "optional") {
- // TODO: Make a review dialog for choosing which ones the user wants!
- if (!had_optional && show_optional_dialog) {
- had_optional = true;
- auto info = CustomMessageBox::selectable(
- m_parent, tr("Optional mod detected!"),
- tr("One or more mods from this modpack are optional. They will be downloaded, but disabled by default!"),
- QMessageBox::Information);
- info->exec();
- }
-
- if (file.path.endsWith(".jar"))
- file.path += ".disabled";
+ file.required = false;
}
}
@@ -385,9 +377,29 @@ bool ModrinthCreationTask::parseManifest(const QString& index_path,
}
}
- files.push_back(file);
+ (file.required ? files : optionalFiles).push_back(file);
}
+ if (!optionalFiles.empty()) {
+ QStringList oFiles;
+ for (auto file : optionalFiles)
+ oFiles.push_back(file.path);
+ OptionalModDialog optionalModDialog(m_parent, oFiles);
+ if (optionalModDialog.exec() == QDialog::Rejected) {
+ emitAborted();
+ return false;
+ }
+
+ auto selectedMods = optionalModDialog.getResult();
+ for (auto file : optionalFiles) {
+ if (selectedMods.contains(file.path)) {
+ file.required = true;
+ } else {
+ file.path += ".disabled";
+ }
+ files.push_back(file);
+ }
+ }
if (set_internal_data) {
auto dependencies = Json::requireObject(obj, "dependencies", "modrinth.index.json");
for (auto it = dependencies.begin(), end = dependencies.end(); it != end; ++it) {
diff --git a/launcher/modplatform/modrinth/ModrinthPackExportTask.cpp b/launcher/modplatform/modrinth/ModrinthPackExportTask.cpp
index a9ddb0c91..e9e8a3b75 100644
--- a/launcher/modplatform/modrinth/ModrinthPackExportTask.cpp
+++ b/launcher/modplatform/modrinth/ModrinthPackExportTask.cpp
@@ -25,6 +25,7 @@
#include "Json.h"
#include "MMCZip.h"
#include "minecraft/PackProfile.h"
+#include "minecraft/mod/MetadataHandler.h"
#include "minecraft/mod/ModFolderModel.h"
const QStringList ModrinthPackExportTask::PREFIXES({ "mods/", "coremods/", "resourcepacks/", "texturepacks/", "shaderpacks/" });
@@ -129,7 +130,8 @@ void ModrinthPackExportTask::collectHashes()
QCryptographicHash sha1(QCryptographicHash::Algorithm::Sha1);
sha1.addData(data);
- ResolvedFile resolvedFile{ sha1.result().toHex(), sha512.result().toHex(), url.toEncoded(), openFile.size() };
+ ResolvedFile resolvedFile{ sha1.result().toHex(), sha512.result().toHex(), url.toEncoded(), openFile.size(),
+ mod->metadata()->side };
resolvedFiles[relative] = resolvedFile;
// nice! we've managed to resolve based on local metadata!
@@ -272,22 +274,33 @@ QByteArray ModrinthPackExportTask::generateIndex()
QString path = iterator.key();
const ResolvedFile& value = iterator.value();
- if (optionalFiles) {
- // detect disabled mod
- const QFileInfo pathInfo(path);
- if (pathInfo.suffix() == "disabled") {
- // rename it
- path = pathInfo.dir().filePath(pathInfo.completeBaseName());
- // ...and make it optional
- QJsonObject env;
- env["client"] = "optional";
- env["server"] = "optional";
- fileOut["env"] = env;
- }
+ QJsonObject env;
+
+ // detect disabled mod
+ const QFileInfo pathInfo(path);
+ if (optionalFiles && pathInfo.suffix() == "disabled") {
+ // rename it
+ path = pathInfo.dir().filePath(pathInfo.completeBaseName());
+ env["client"] = "optional";
+ env["server"] = "optional";
+ } else {
+ env["client"] = "required";
+ env["server"] = "required";
+ }
+ switch (iterator->side) {
+ case Metadata::ModSide::ClientSide:
+ env["server"] = "unsupported";
+ break;
+ case Metadata::ModSide::ServerSide:
+ env["client"] = "unsupported";
+ break;
+ case Metadata::ModSide::UniversalSide:
+ break;
}
+ fileOut["env"] = env;
fileOut["path"] = path;
- fileOut["downloads"] = QJsonArray{ iterator.value().url };
+ fileOut["downloads"] = QJsonArray{ iterator->url };
QJsonObject hashes;
hashes["sha1"] = value.sha1;
diff --git a/launcher/modplatform/modrinth/ModrinthPackExportTask.h b/launcher/modplatform/modrinth/ModrinthPackExportTask.h
index 83540dfac..33c42e817 100644
--- a/launcher/modplatform/modrinth/ModrinthPackExportTask.h
+++ b/launcher/modplatform/modrinth/ModrinthPackExportTask.h
@@ -44,6 +44,7 @@ class ModrinthPackExportTask : public Task {
struct ResolvedFile {
QString sha1, sha512, url;
qint64 size;
+ Metadata::ModSide side;
};
static const QStringList PREFIXES;
diff --git a/launcher/modplatform/modrinth/ModrinthPackIndex.cpp b/launcher/modplatform/modrinth/ModrinthPackIndex.cpp
index 107b99006..7d0893261 100644
--- a/launcher/modplatform/modrinth/ModrinthPackIndex.cpp
+++ b/launcher/modplatform/modrinth/ModrinthPackIndex.cpp
@@ -27,6 +27,11 @@
static ModrinthAPI api;
static ModPlatform::ProviderCapabilities ProviderCaps;
+bool shouldDownloadOnSide(QString side)
+{
+ return side == "required" || side == "optional";
+}
+
// https://docs.modrinth.com/api-spec/#tag/projects/operation/getProject
void Modrinth::loadIndexedPack(ModPlatform::IndexedPack& pack, QJsonObject& obj)
{
@@ -53,6 +58,17 @@ void Modrinth::loadIndexedPack(ModPlatform::IndexedPack& pack, QJsonObject& obj)
modAuthor.url = api.getAuthorURL(modAuthor.name);
pack.authors.append(modAuthor);
+ auto client = shouldDownloadOnSide(Json::ensureString(obj, "client_side"));
+ auto server = shouldDownloadOnSide(Json::ensureString(obj, "server_side"));
+
+ if (server && client) {
+ pack.side = "both";
+ } else if (server) {
+ pack.side = "server";
+ } else if (client) {
+ pack.side = "client";
+ }
+
// Modrinth can have more data than what's provided by the basic search :)
pack.extraDataLoaded = false;
}
@@ -149,6 +165,8 @@ auto Modrinth::loadIndexedPackVersion(QJsonObject& obj, QString preferred_hash_t
}
file.version = Json::requireString(obj, "name");
file.version_number = Json::requireString(obj, "version_number");
+ file.version_type = ModPlatform::IndexedVersionType(Json::requireString(obj, "version_type"));
+
file.changelog = Json::requireString(obj, "changelog");
auto dependencies = Json::ensureArray(obj, "dependencies");
diff --git a/launcher/modplatform/modrinth/ModrinthPackManifest.cpp b/launcher/modplatform/modrinth/ModrinthPackManifest.cpp
index 0d07c6361..d86f1c0e5 100644
--- a/launcher/modplatform/modrinth/ModrinthPackManifest.cpp
+++ b/launcher/modplatform/modrinth/ModrinthPackManifest.cpp
@@ -35,6 +35,7 @@
*/
#include "ModrinthPackManifest.h"
+#include
#include "Json.h"
#include "modplatform/modrinth/ModrinthAPI.h"
@@ -56,8 +57,8 @@ void loadIndexedPack(Modpack& pack, QJsonObject& obj)
pack.description = Json::ensureString(obj, "description");
auto temp_author_name = Json::ensureString(obj, "author");
pack.author = std::make_tuple(temp_author_name, api.getAuthorURL(temp_author_name));
- pack.iconName = QString("modrinth_%1").arg(Json::ensureString(obj, "slug"));
pack.iconUrl = Json::ensureString(obj, "icon_url");
+ pack.iconName = QString("modrinth_%1.%2").arg(Json::ensureString(obj, "slug"), QFileInfo(pack.iconUrl.fileName()).suffix());
}
void loadIndexedInfo(Modpack& pack, QJsonObject& obj)
@@ -128,6 +129,7 @@ auto loadIndexedVersion(QJsonObject& obj) -> ModpackVersion
file.name = Json::requireString(obj, "name");
file.version = Json::requireString(obj, "version_number");
+ file.version_type = ModPlatform::IndexedVersionType(Json::requireString(obj, "version_type"));
file.changelog = Json::ensureString(obj, "changelog");
file.id = Json::requireString(obj, "id");
diff --git a/launcher/modplatform/modrinth/ModrinthPackManifest.h b/launcher/modplatform/modrinth/ModrinthPackManifest.h
index effa1a84a..6fc55a043 100644
--- a/launcher/modplatform/modrinth/ModrinthPackManifest.h
+++ b/launcher/modplatform/modrinth/ModrinthPackManifest.h
@@ -45,6 +45,8 @@
#include
#include
+#include "modplatform/ModIndex.h"
+
class MinecraftInstance;
namespace Modrinth {
@@ -55,6 +57,7 @@ struct File {
QCryptographicHash::Algorithm hashAlgorithm;
QByteArray hash;
QQueue downloads;
+ bool required = true;
};
struct DonationData {
@@ -79,6 +82,7 @@ struct ModpackExtra {
struct ModpackVersion {
QString name;
QString version;
+ ModPlatform::IndexedVersionType version_type;
QString changelog;
QString id;
diff --git a/launcher/modplatform/packwiz/Packwiz.cpp b/launcher/modplatform/packwiz/Packwiz.cpp
index 71f66bf3e..e35567f24 100644
--- a/launcher/modplatform/packwiz/Packwiz.cpp
+++ b/launcher/modplatform/packwiz/Packwiz.cpp
@@ -21,6 +21,8 @@
#include
#include
#include
+#include
+#include
#include "FileSystem.h"
#include "StringUtils.h"
@@ -111,6 +113,7 @@ auto V1::createModFormat([[maybe_unused]] QDir& index_dir, ModPlatform::IndexedP
mod.provider = mod_pack.provider;
mod.file_id = mod_version.fileId;
mod.project_id = mod_pack.addonId;
+ mod.side = stringToSide(mod_pack.side);
return mod;
}
@@ -154,38 +157,52 @@ void V1::updateModIndex(QDir& index_dir, Mod& mod)
FS::ensureFilePathExists(index_file.fileName());
}
+ toml::table update;
+ switch (mod.provider) {
+ case (ModPlatform::ResourceProvider::FLAME):
+ if (mod.file_id.toInt() == 0 || mod.project_id.toInt() == 0) {
+ qCritical() << QString("Did not write file %1 because missing information!").arg(normalized_fname);
+ return;
+ }
+ update = toml::table{
+ { "file-id", mod.file_id.toInt() },
+ { "project-id", mod.project_id.toInt() },
+ };
+ break;
+ case (ModPlatform::ResourceProvider::MODRINTH):
+ if (mod.mod_id().toString().isEmpty() || mod.version().toString().isEmpty()) {
+ qCritical() << QString("Did not write file %1 because missing information!").arg(normalized_fname);
+ return;
+ }
+ update = toml::table{
+ { "mod-id", mod.mod_id().toString().toStdString() },
+ { "version", mod.version().toString().toStdString() },
+ };
+ break;
+ }
+
if (!index_file.open(QIODevice::ReadWrite)) {
- qCritical() << QString("Could not open file %1!").arg(indexFileName(mod.name));
+ qCritical() << QString("Could not open file %1!").arg(normalized_fname);
return;
}
// Put TOML data into the file
QTextStream in_stream(&index_file);
- auto addToStream = [&in_stream](QString&& key, QString value) { in_stream << QString("%1 = \"%2\"\n").arg(key, value); };
-
{
- addToStream("name", mod.name);
- addToStream("filename", mod.filename);
- addToStream("side", mod.side);
-
- in_stream << QString("\n[download]\n");
- addToStream("mode", mod.mode);
- addToStream("url", mod.url.toString());
- addToStream("hash-format", mod.hash_format);
- addToStream("hash", mod.hash);
-
- in_stream << QString("\n[update]\n");
- in_stream << QString("[update.%1]\n").arg(ProviderCaps.name(mod.provider));
- switch (mod.provider) {
- case (ModPlatform::ResourceProvider::FLAME):
- in_stream << QString("file-id = %1\n").arg(mod.file_id.toString());
- in_stream << QString("project-id = %1\n").arg(mod.project_id.toString());
- break;
- case (ModPlatform::ResourceProvider::MODRINTH):
- addToStream("mod-id", mod.mod_id().toString());
- addToStream("version", mod.version().toString());
- break;
- }
+ auto tbl = toml::table{ { "name", mod.name.toStdString() },
+ { "filename", mod.filename.toStdString() },
+ { "side", sideToString(mod.side).toStdString() },
+ { "download",
+ toml::table{
+ { "mode", mod.mode.toStdString() },
+ { "url", mod.url.toString().toStdString() },
+ { "hash-format", mod.hash_format.toStdString() },
+ { "hash", mod.hash.toStdString() },
+ } },
+ { "update", toml::table{ { ProviderCaps.name(mod.provider), update } } } };
+ std::stringstream ss;
+ ss << tbl;
+ in_stream << QString::fromStdString(ss.str());
}
index_file.flush();
@@ -258,7 +275,7 @@ auto V1::getIndexForMod(QDir& index_dir, QString slug) -> Mod
{ // Basic info
mod.name = stringEntry(table, "name");
mod.filename = stringEntry(table, "filename");
- mod.side = stringEntry(table, "side");
+ mod.side = stringToSide(stringEntry(table, "side"));
}
{ // [download] info
@@ -313,4 +330,28 @@ auto V1::getIndexForMod(QDir& index_dir, QVariant& mod_id) -> Mod
return {};
}
+auto V1::sideToString(Side side) -> QString
+{
+ switch (side) {
+ case Side::ClientSide:
+ return "client";
+ case Side::ServerSide:
+ return "server";
+ case Side::UniversalSide:
+ return "both";
+ }
+ return {};
+}
+
+auto V1::stringToSide(QString side) -> Side
+{
+ if (side == "client")
+ return Side::ClientSide;
+ if (side == "server")
+ return Side::ServerSide;
+ if (side == "both")
+ return Side::UniversalSide;
+ return Side::UniversalSide;
+}
+
} // namespace Packwiz
diff --git a/launcher/modplatform/packwiz/Packwiz.h b/launcher/modplatform/packwiz/Packwiz.h
index 7edc18cde..dce198b0e 100644
--- a/launcher/modplatform/packwiz/Packwiz.h
+++ b/launcher/modplatform/packwiz/Packwiz.h
@@ -35,12 +35,12 @@ auto getRealIndexName(QDir& index_dir, QString normalized_index_name, bool shoul
class V1 {
public:
+ enum class Side { ClientSide = 1 << 0, ServerSide = 1 << 1, UniversalSide = ClientSide | ServerSide };
struct Mod {
QString slug{};
QString name{};
QString filename{};
- // FIXME: make side an enum
- QString side{ "both" };
+ Side side{ Side::UniversalSide };
// [download]
QString mode{};
@@ -93,6 +93,9 @@ class V1 {
* If the mod doesn't have a metadata, it simply returns an empty Mod object.
* */
static auto getIndexForMod(QDir& index_dir, QVariant& mod_id) -> Mod;
+
+ static auto sideToString(Side side) -> QString;
+ static auto stringToSide(QString side) -> Side;
};
} // namespace Packwiz
diff --git a/launcher/net/Download.cpp b/launcher/net/Download.cpp
index d25447b2d..bae364f12 100644
--- a/launcher/net/Download.cpp
+++ b/launcher/net/Download.cpp
@@ -89,4 +89,4 @@ QNetworkReply* Download::getReply(QNetworkRequest& request)
{
return m_network->get(request);
}
-} // namespace Net
\ No newline at end of file
+} // namespace Net
diff --git a/launcher/net/NetJob.cpp b/launcher/net/NetJob.cpp
index 3869316e3..784d81c37 100644
--- a/launcher/net/NetJob.cpp
+++ b/launcher/net/NetJob.cpp
@@ -36,6 +36,16 @@
*/
#include "NetJob.h"
+#if defined(LAUNCHER_APPLICATION)
+#include "Application.h"
+#endif
+
+NetJob::NetJob(QString job_name, shared_qobject_ptr network) : ConcurrentTask(nullptr, job_name), m_network(network)
+{
+#if defined(LAUNCHER_APPLICATION)
+ setMaxConcurrent(APPLICATION->settings()->get("NumberOfConcurrentDownloads").toInt());
+#endif
+}
auto NetJob::addNetAction(NetAction::Ptr action) -> bool
{
diff --git a/launcher/net/NetJob.h b/launcher/net/NetJob.h
index cc63f4497..1c4337ec6 100644
--- a/launcher/net/NetJob.h
+++ b/launcher/net/NetJob.h
@@ -52,9 +52,7 @@ class NetJob : public ConcurrentTask {
public:
using Ptr = shared_qobject_ptr;
- explicit NetJob(QString job_name, shared_qobject_ptr network)
- : ConcurrentTask(nullptr, job_name), m_network(network)
- {}
+ explicit NetJob(QString job_name, shared_qobject_ptr network);
~NetJob() override = default;
void startNext() override;
diff --git a/launcher/net/NetRequest.cpp b/launcher/net/NetRequest.cpp
index ff59da18b..728c0e077 100644
--- a/launcher/net/NetRequest.cpp
+++ b/launcher/net/NetRequest.cpp
@@ -46,6 +46,7 @@
#if defined(LAUNCHER_APPLICATION)
#include "Application.h"
#endif
+#include "BuildConfig.h"
#include "net/NetAction.h"
@@ -111,6 +112,8 @@ void NetRequest::executeTask()
m_last_progress_bytes = 0;
QNetworkReply* rep = getReply(request);
+ if (rep == nullptr) // it failed
+ return;
m_reply.reset(rep);
connect(rep, &QNetworkReply::downloadProgress, this, &NetRequest::downloadProgress);
connect(rep, &QNetworkReply::finished, this, &NetRequest::downloadFinished);
diff --git a/launcher/net/NetRequest.h b/launcher/net/NetRequest.h
index ee47ab2a6..0b307b4f6 100644
--- a/launcher/net/NetRequest.h
+++ b/launcher/net/NetRequest.h
@@ -87,7 +87,7 @@ class NetRequest : public NetAction {
std::unique_ptr m_sink;
Options m_options;
- typedef const QLoggingCategory& (*logCatFunc)();
+ using logCatFunc = const QLoggingCategory& (*)();
logCatFunc logCat = taskUploadLogC;
std::chrono::steady_clock m_clock;
diff --git a/launcher/net/StaticHeaderProxy.h b/launcher/net/StaticHeaderProxy.h
new file mode 100644
index 000000000..8af7d203d
--- /dev/null
+++ b/launcher/net/StaticHeaderProxy.h
@@ -0,0 +1,39 @@
+// SPDX-License-Identifier: GPL-3.0-only
+/*
+ * Prism Launcher - Minecraft Launcher
+ * Copyright (C) 2023 Rachel Powers <508861+Ryex@users.noreply.github.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 .
+ *
+ */
+
+#pragma once
+
+#include "net/HeaderProxy.h"
+
+namespace Net {
+
+class StaticHeaderProxy : public HeaderProxy {
+ public:
+ StaticHeaderProxy(QList hdrs = {}) : HeaderProxy(), m_hdrs(hdrs){};
+ virtual ~StaticHeaderProxy() = default;
+
+ public:
+ virtual QList headers(const QNetworkRequest&) const override { return m_hdrs; };
+ void setHeaders(QList hdrs) { m_hdrs = hdrs; };
+
+ private:
+ QList m_hdrs;
+};
+
+} // namespace Net
diff --git a/launcher/news/NewsEntry.h b/launcher/news/NewsEntry.h
index 2d409a9fb..ab717ec89 100644
--- a/launcher/news/NewsEntry.h
+++ b/launcher/news/NewsEntry.h
@@ -51,4 +51,4 @@ class NewsEntry : public QObject {
QString link;
};
-typedef std::shared_ptr NewsEntryPtr;
+using NewsEntryPtr = std::shared_ptr;
diff --git a/launcher/pathmatcher/IPathMatcher.h b/launcher/pathmatcher/IPathMatcher.h
index cd6121979..f3b01e8cf 100644
--- a/launcher/pathmatcher/IPathMatcher.h
+++ b/launcher/pathmatcher/IPathMatcher.h
@@ -4,7 +4,7 @@
class IPathMatcher {
public:
- typedef std::shared_ptr Ptr;
+ using Ptr = std::shared_ptr;
public:
virtual ~IPathMatcher() {}
diff --git a/launcher/resources/breeze_dark/breeze_dark.qrc b/launcher/resources/breeze_dark/breeze_dark.qrc
index 320ca8171..61d82ec30 100644
--- a/launcher/resources/breeze_dark/breeze_dark.qrc
+++ b/launcher/resources/breeze_dark/breeze_dark.qrc
@@ -9,6 +9,7 @@
scalable/copy.svg
scalable/coremods.svg
scalable/custom-commands.svg
+ scalable/environment-variables.svg
scalable/discord.svg
scalable/externaltools.svg
scalable/help.svg
diff --git a/launcher/resources/breeze_dark/scalable/environment-variables.svg b/launcher/resources/breeze_dark/scalable/environment-variables.svg
new file mode 100644
index 000000000..308c4a239
--- /dev/null
+++ b/launcher/resources/breeze_dark/scalable/environment-variables.svg
@@ -0,0 +1,13 @@
+
diff --git a/launcher/resources/breeze_light/breeze_light.qrc b/launcher/resources/breeze_light/breeze_light.qrc
index e88cd9a00..2211c7188 100644
--- a/launcher/resources/breeze_light/breeze_light.qrc
+++ b/launcher/resources/breeze_light/breeze_light.qrc
@@ -9,6 +9,7 @@
scalable/copy.svg
scalable/coremods.svg
scalable/custom-commands.svg
+ scalable/environment-variables.svg
scalable/discord.svg
scalable/externaltools.svg
scalable/help.svg
diff --git a/launcher/resources/breeze_light/scalable/environment-variables.svg b/launcher/resources/breeze_light/scalable/environment-variables.svg
new file mode 100644
index 000000000..f5d4acc3b
--- /dev/null
+++ b/launcher/resources/breeze_light/scalable/environment-variables.svg
@@ -0,0 +1,13 @@
+
diff --git a/launcher/resources/flat/flat.qrc b/launcher/resources/flat/flat.qrc
index 2fd5daefe..8876027da 100644
--- a/launcher/resources/flat/flat.qrc
+++ b/launcher/resources/flat/flat.qrc
@@ -11,6 +11,7 @@
scalable/copy.svg
scalable/coremods.svg
scalable/custom-commands.svg
+ scalable/environment-variables.svg
scalable/discord.svg
scalable/externaltools.svg
scalable/help.svg
diff --git a/launcher/resources/flat/scalable/custom-commands.svg b/launcher/resources/flat/scalable/custom-commands.svg
index a35634b16..f2e587843 100644
--- a/launcher/resources/flat/scalable/custom-commands.svg
+++ b/launcher/resources/flat/scalable/custom-commands.svg
@@ -1,86 +1 @@
-
-
+
diff --git a/launcher/resources/flat/scalable/environment-variables.svg b/launcher/resources/flat/scalable/environment-variables.svg
new file mode 100644
index 000000000..a35634b16
--- /dev/null
+++ b/launcher/resources/flat/scalable/environment-variables.svg
@@ -0,0 +1,86 @@
+
+
diff --git a/launcher/resources/flat_white/flat_white.qrc b/launcher/resources/flat_white/flat_white.qrc
index a1c940da0..83b178cbf 100644
--- a/launcher/resources/flat_white/flat_white.qrc
+++ b/launcher/resources/flat_white/flat_white.qrc
@@ -11,6 +11,7 @@
scalable/copy.svg
scalable/coremods.svg
scalable/custom-commands.svg
+ scalable/environment-variables.svg
scalable/discord.svg
scalable/externaltools.svg
scalable/help.svg
diff --git a/launcher/resources/flat_white/scalable/custom-commands.svg b/launcher/resources/flat_white/scalable/custom-commands.svg
index fe1cf9987..0ba459cff 100644
--- a/launcher/resources/flat_white/scalable/custom-commands.svg
+++ b/launcher/resources/flat_white/scalable/custom-commands.svg
@@ -1,86 +1 @@
-
-
+
diff --git a/launcher/resources/flat_white/scalable/environment-variables.svg b/launcher/resources/flat_white/scalable/environment-variables.svg
new file mode 100644
index 000000000..fe1cf9987
--- /dev/null
+++ b/launcher/resources/flat_white/scalable/environment-variables.svg
@@ -0,0 +1,86 @@
+
+
diff --git a/launcher/resources/multimc/multimc.qrc b/launcher/resources/multimc/multimc.qrc
index 80981559b..eeba32186 100644
--- a/launcher/resources/multimc/multimc.qrc
+++ b/launcher/resources/multimc/multimc.qrc
@@ -73,8 +73,8 @@
64x64/screenshots.png
scalable/screenshots.svg
-
scalable/custom-commands.svg
+ scalable/environment-variables.svg
16x16/cat.png
diff --git a/launcher/resources/multimc/scalable/custom-commands.svg b/launcher/resources/multimc/scalable/custom-commands.svg
index b7f1a149b..0d502bb1d 100644
--- a/launcher/resources/multimc/scalable/custom-commands.svg
+++ b/launcher/resources/multimc/scalable/custom-commands.svg
@@ -1,338 +1,4339 @@
+
diff --git a/launcher/resources/multimc/scalable/environment-variables.svg b/launcher/resources/multimc/scalable/environment-variables.svg
new file mode 100644
index 000000000..5e136b202
--- /dev/null
+++ b/launcher/resources/multimc/scalable/environment-variables.svg
@@ -0,0 +1,346 @@
+
+
+
+
diff --git a/launcher/resources/pe_blue/pe_blue.qrc b/launcher/resources/pe_blue/pe_blue.qrc
index da45ef9a1..717d3972e 100644
--- a/launcher/resources/pe_blue/pe_blue.qrc
+++ b/launcher/resources/pe_blue/pe_blue.qrc
@@ -10,6 +10,7 @@
scalable/copy.svg
scalable/coremods.svg
scalable/custom-commands.svg
+ scalable/environment-variables.svg
scalable/externaltools.svg
scalable/help.svg
scalable/instance-settings.svg
diff --git a/launcher/resources/pe_blue/scalable/environment-variables.svg b/launcher/resources/pe_blue/scalable/environment-variables.svg
new file mode 100644
index 000000000..61c63a4c5
--- /dev/null
+++ b/launcher/resources/pe_blue/scalable/environment-variables.svg
@@ -0,0 +1,345 @@
+
+
diff --git a/launcher/resources/pe_colored/pe_colored.qrc b/launcher/resources/pe_colored/pe_colored.qrc
index ba5bd44f9..023c81e74 100644
--- a/launcher/resources/pe_colored/pe_colored.qrc
+++ b/launcher/resources/pe_colored/pe_colored.qrc
@@ -10,6 +10,7 @@
scalable/copy.svg
scalable/coremods.svg
scalable/custom-commands.svg
+ scalable/environment-variables.svg
scalable/externaltools.svg
scalable/help.svg
scalable/instance-settings.svg
diff --git a/launcher/resources/pe_colored/scalable/environment-variables.svg b/launcher/resources/pe_colored/scalable/environment-variables.svg
new file mode 100644
index 000000000..c1aab6bca
--- /dev/null
+++ b/launcher/resources/pe_colored/scalable/environment-variables.svg
@@ -0,0 +1,347 @@
+
+
diff --git a/launcher/resources/pe_dark/pe_dark.qrc b/launcher/resources/pe_dark/pe_dark.qrc
index 2bfec42cb..c97fb469c 100644
--- a/launcher/resources/pe_dark/pe_dark.qrc
+++ b/launcher/resources/pe_dark/pe_dark.qrc
@@ -10,6 +10,7 @@
scalable/copy.svg
scalable/coremods.svg
scalable/custom-commands.svg
+ scalable/environment-variables.svg
scalable/externaltools.svg
scalable/help.svg
scalable/instance-settings.svg
diff --git a/launcher/resources/pe_dark/scalable/environment-variables.svg b/launcher/resources/pe_dark/scalable/environment-variables.svg
new file mode 100644
index 000000000..46a3445f4
--- /dev/null
+++ b/launcher/resources/pe_dark/scalable/environment-variables.svg
@@ -0,0 +1,345 @@
+
+
diff --git a/launcher/resources/pe_light/pe_light.qrc b/launcher/resources/pe_light/pe_light.qrc
index 25d5da73b..b590dd2c6 100644
--- a/launcher/resources/pe_light/pe_light.qrc
+++ b/launcher/resources/pe_light/pe_light.qrc
@@ -10,6 +10,7 @@
scalable/copy.svg
scalable/coremods.svg
scalable/custom-commands.svg
+ scalable/environment-variables.svg
scalable/externaltools.svg
scalable/help.svg
scalable/instance-settings.svg
diff --git a/launcher/resources/pe_light/scalable/environment-variables.svg b/launcher/resources/pe_light/scalable/environment-variables.svg
new file mode 100644
index 000000000..b8d562ffe
--- /dev/null
+++ b/launcher/resources/pe_light/scalable/environment-variables.svg
@@ -0,0 +1,345 @@
+
+
diff --git a/launcher/screenshots/ImgurAlbumCreation.cpp b/launcher/screenshots/ImgurAlbumCreation.cpp
index d548c89ae..7e42ff40c 100644
--- a/launcher/screenshots/ImgurAlbumCreation.cpp
+++ b/launcher/screenshots/ImgurAlbumCreation.cpp
@@ -39,87 +39,76 @@
#include
#include
#include
+#include
#include
#include
#include
+#include
-#include "Application.h"
#include "BuildConfig.h"
+#include "net/StaticHeaderProxy.h"
-ImgurAlbumCreation::ImgurAlbumCreation(QList screenshots) : NetAction(), m_screenshots(screenshots)
+Net::NetRequest::Ptr ImgurAlbumCreation::make(std::shared_ptr output, QList screenshots)
{
- m_url = BuildConfig.IMGUR_BASE_URL + "album.json";
- m_state = State::Inactive;
+ auto up = makeShared();
+ up->m_url = BuildConfig.IMGUR_BASE_URL + "album.json";
+ up->m_sink.reset(new Sink(output));
+ up->m_screenshots = screenshots;
+ return up;
}
-void ImgurAlbumCreation::executeTask()
+QNetworkReply* ImgurAlbumCreation::getReply(QNetworkRequest& request)
{
- m_state = State::Running;
- QNetworkRequest request(m_url);
- request.setHeader(QNetworkRequest::UserAgentHeader, APPLICATION->getUserAgentUncached().toUtf8());
- request.setHeader(QNetworkRequest::ContentTypeHeader, "application/x-www-form-urlencoded");
- request.setRawHeader("Authorization", QString("Client-ID %1").arg(BuildConfig.IMGUR_CLIENT_ID).toStdString().c_str());
- request.setRawHeader("Accept", "application/json");
-
QStringList hashes;
for (auto shot : m_screenshots) {
hashes.append(shot->m_imgurDeleteHash);
}
-
const QByteArray data = "deletehashes=" + hashes.join(',').toUtf8() + "&title=Minecraft%20Screenshots&privacy=hidden";
+ return m_network->post(request, data);
+};
- QNetworkReply* rep = APPLICATION->network()->post(request, data);
-
- m_reply.reset(rep);
- connect(rep, &QNetworkReply::uploadProgress, this, &ImgurAlbumCreation::downloadProgress);
- connect(rep, &QNetworkReply::finished, this, &ImgurAlbumCreation::downloadFinished);
-#if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0) // QNetworkReply::errorOccurred added in 5.15
- connect(rep, &QNetworkReply::errorOccurred, this, &ImgurAlbumCreation::downloadError);
-#else
- connect(rep, QOverload::of(&QNetworkReply::error), this, &ImgurAlbumCreation::downloadError);
-#endif
- connect(rep, &QNetworkReply::sslErrors, this, &ImgurAlbumCreation::sslErrors);
+void ImgurAlbumCreation::init()
+{
+ qDebug() << "Setting up imgur upload";
+ auto api_headers = new Net::StaticHeaderProxy(
+ QList{ { "Content-Type", "application/x-www-form-urlencoded" },
+ { "Authorization", QString("Client-ID %1").arg(BuildConfig.IMGUR_CLIENT_ID).toStdString().c_str() },
+ { "Accept", "application/json" } });
+ addHeaderProxy(api_headers);
}
-void ImgurAlbumCreation::downloadError([[maybe_unused]] QNetworkReply::NetworkError error)
+auto ImgurAlbumCreation::Sink::init(QNetworkRequest& request) -> Task::State
{
- qDebug() << m_reply->errorString();
- m_state = State::Failed;
-}
+ m_output.clear();
+ return Task::State::Running;
+};
-void ImgurAlbumCreation::downloadFinished()
+auto ImgurAlbumCreation::Sink::write(QByteArray& data) -> Task::State
{
- if (m_state != State::Failed) {
- QByteArray data = m_reply->readAll();
- m_reply.reset();
- QJsonParseError jsonError;
- QJsonDocument doc = QJsonDocument::fromJson(data, &jsonError);
- if (jsonError.error != QJsonParseError::NoError) {
- qDebug() << jsonError.errorString();
- emitFailed();
- return;
- }
- auto object = doc.object();
- if (!object.value("success").toBool()) {
- qDebug() << doc.toJson();
- emitFailed();
- return;
- }
- m_deleteHash = object.value("data").toObject().value("deletehash").toString();
- m_id = object.value("data").toObject().value("id").toString();
- m_state = State::Succeeded;
- emit succeeded();
- return;
- } else {
- qDebug() << m_reply->readAll();
- m_reply.reset();
- emitFailed();
- return;
- }
+ m_output.append(data);
+ return Task::State::Running;
}
-void ImgurAlbumCreation::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
+auto ImgurAlbumCreation::Sink::abort() -> Task::State
{
- setProgress(bytesReceived, bytesTotal);
- emit progress(bytesReceived, bytesTotal);
+ m_output.clear();
+ return Task::State::Failed;
}
+
+auto ImgurAlbumCreation::Sink::finalize(QNetworkReply&) -> Task::State
+{
+ QJsonParseError jsonError;
+ QJsonDocument doc = QJsonDocument::fromJson(m_output, &jsonError);
+ if (jsonError.error != QJsonParseError::NoError) {
+ qDebug() << jsonError.errorString();
+ return Task::State::Failed;
+ }
+ auto object = doc.object();
+ if (!object.value("success").toBool()) {
+ qDebug() << doc.toJson();
+ return Task::State::Failed;
+ }
+ m_result->deleteHash = object.value("data").toObject().value("deletehash").toString();
+ m_result->id = object.value("data").toObject().value("id").toString();
+ return Task::State::Succeeded;
+}
\ No newline at end of file
diff --git a/launcher/screenshots/ImgurAlbumCreation.h b/launcher/screenshots/ImgurAlbumCreation.h
index bb71bec2b..7c292db73 100644
--- a/launcher/screenshots/ImgurAlbumCreation.h
+++ b/launcher/screenshots/ImgurAlbumCreation.h
@@ -36,34 +36,39 @@
#pragma once
#include "Screenshot.h"
-#include "net/NetAction.h"
+#include "net/NetRequest.h"
-typedef shared_qobject_ptr ImgurAlbumCreationPtr;
-class ImgurAlbumCreation : public NetAction {
+class ImgurAlbumCreation : public Net::NetRequest {
public:
- explicit ImgurAlbumCreation(QList screenshots);
- static ImgurAlbumCreationPtr make(QList screenshots)
- {
- return ImgurAlbumCreationPtr(new ImgurAlbumCreation(screenshots));
- }
+ virtual ~ImgurAlbumCreation() = default;
- QString deleteHash() const { return m_deleteHash; }
- QString id() const { return m_id; }
+ struct Result {
+ QString deleteHash;
+ QString id;
+ };
- void init() override{};
+ class Sink : public Net::Sink {
+ public:
+ Sink(std::shared_ptr res) : m_result(res){};
+ virtual ~Sink() = default;
- protected slots:
- void downloadProgress(qint64 bytesReceived, qint64 bytesTotal) override;
- void downloadError(QNetworkReply::NetworkError error) override;
- void downloadFinished() override;
- void downloadReadyRead() override {}
+ public:
+ auto init(QNetworkRequest& request) -> Task::State override;
+ auto write(QByteArray& data) -> Task::State override;
+ auto abort() -> Task::State override;
+ auto finalize(QNetworkReply& reply) -> Task::State override;
+ auto hasLocalData() -> bool override { return false; }
- public slots:
- void executeTask() override;
+ private:
+ std::shared_ptr m_result;
+ QByteArray m_output;
+ };
+
+ static NetRequest::Ptr make(std::shared_ptr output, QList screenshots);
+ QNetworkReply* getReply(QNetworkRequest& request) override;
+
+ void init() override;
private:
QList m_screenshots;
-
- QString m_deleteHash;
- QString m_id;
};
diff --git a/launcher/screenshots/ImgurUpload.cpp b/launcher/screenshots/ImgurUpload.cpp
index 29499a20d..7ed672eb7 100644
--- a/launcher/screenshots/ImgurUpload.cpp
+++ b/launcher/screenshots/ImgurUpload.cpp
@@ -35,8 +35,8 @@
*/
#include "ImgurUpload.h"
-#include "Application.h"
#include "BuildConfig.h"
+#include "net/StaticHeaderProxy.h"
#include
#include
@@ -47,104 +47,84 @@
#include
#include
-ImgurUpload::ImgurUpload(ScreenShot::Ptr shot) : NetAction(), m_shot(shot)
+void ImgurUpload::init()
{
- m_url = BuildConfig.IMGUR_BASE_URL + "upload.json";
- m_state = State::Inactive;
+ qDebug() << "Setting up imgur upload";
+ auto api_headers = new Net::StaticHeaderProxy(
+ QList{ { "Authorization", QString("Client-ID %1").arg(BuildConfig.IMGUR_CLIENT_ID).toStdString().c_str() },
+ { "Accept", "application/json" } });
+ addHeaderProxy(api_headers);
}
-void ImgurUpload::executeTask()
+QNetworkReply* ImgurUpload::getReply(QNetworkRequest& request)
{
- finished = false;
- m_state = Task::State::Running;
- QNetworkRequest request(m_url);
- request.setHeader(QNetworkRequest::UserAgentHeader, APPLICATION->getUserAgentUncached().toUtf8());
- request.setRawHeader("Authorization", QString("Client-ID %1").arg(BuildConfig.IMGUR_CLIENT_ID).toStdString().c_str());
- request.setRawHeader("Accept", "application/json");
+ auto file = new QFile(m_fileInfo.absoluteFilePath());
- QFile f(m_shot->m_file.absoluteFilePath());
- if (!f.open(QFile::ReadOnly)) {
+ if (!file->open(QFile::ReadOnly)) {
emitFailed();
- return;
+ return nullptr;
}
QHttpMultiPart* multipart = new QHttpMultiPart(QHttpMultiPart::FormDataType);
+ file->setParent(multipart);
QHttpPart filePart;
- filePart.setBody(f.readAll().toBase64());
+ filePart.setBodyDevice(file);
filePart.setHeader(QNetworkRequest::ContentTypeHeader, "image/png");
filePart.setHeader(QNetworkRequest::ContentDispositionHeader, "form-data; name=\"image\"");
multipart->append(filePart);
QHttpPart typePart;
typePart.setHeader(QNetworkRequest::ContentDispositionHeader, "form-data; name=\"type\"");
- typePart.setBody("base64");
+ typePart.setBody("file");
multipart->append(typePart);
QHttpPart namePart;
namePart.setHeader(QNetworkRequest::ContentDispositionHeader, "form-data; name=\"name\"");
- namePart.setBody(m_shot->m_file.baseName().toUtf8());
+ namePart.setBody(m_fileInfo.baseName().toUtf8());
multipart->append(namePart);
- QNetworkReply* rep = m_network->post(request, multipart);
+ return m_network->post(request, multipart);
+};
- m_reply.reset(rep);
- connect(rep, &QNetworkReply::uploadProgress, this, &ImgurUpload::downloadProgress);
- connect(rep, &QNetworkReply::finished, this, &ImgurUpload::downloadFinished);
-#if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0) // QNetworkReply::errorOccurred added in 5.15
- connect(rep, &QNetworkReply::errorOccurred, this, &ImgurUpload::downloadError);
-#else
- connect(rep, QOverload::of(&QNetworkReply::error), this, &ImgurUpload::downloadError);
-#endif
- connect(rep, &QNetworkReply::sslErrors, this, &ImgurUpload::sslErrors);
+auto ImgurUpload::Sink::init(QNetworkRequest& request) -> Task::State
+{
+ m_output.clear();
+ return Task::State::Running;
+};
+
+auto ImgurUpload::Sink::write(QByteArray& data) -> Task::State
+{
+ m_output.append(data);
+ return Task::State::Running;
}
-void ImgurUpload::downloadError([[maybe_unused]] QNetworkReply::NetworkError error)
+auto ImgurUpload::Sink::abort() -> Task::State
{
- qCritical() << "ImgurUpload failed with error" << m_reply->errorString() << "Server reply:\n" << m_reply->readAll();
- if (finished) {
- qCritical() << "Double finished ImgurUpload!";
- return;
- }
- m_state = Task::State::Failed;
- finished = true;
- m_reply.reset();
- emitFailed();
+ m_output.clear();
+ return Task::State::Failed;
}
-void ImgurUpload::downloadFinished()
+auto ImgurUpload::Sink::finalize(QNetworkReply&) -> Task::State
{
- if (finished) {
- qCritical() << "Double finished ImgurUpload!";
- return;
- }
- QByteArray data = m_reply->readAll();
- m_reply.reset();
QJsonParseError jsonError;
- QJsonDocument doc = QJsonDocument::fromJson(data, &jsonError);
+ QJsonDocument doc = QJsonDocument::fromJson(m_output, &jsonError);
if (jsonError.error != QJsonParseError::NoError) {
qDebug() << "imgur server did not reply with JSON" << jsonError.errorString();
- finished = true;
- m_reply.reset();
- emitFailed();
- return;
+ return Task::State::Failed;
}
auto object = doc.object();
if (!object.value("success").toBool()) {
qDebug() << "Screenshot upload not successful:" << doc.toJson();
- finished = true;
- m_reply.reset();
- emitFailed();
- return;
+ return Task::State::Failed;
}
m_shot->m_imgurId = object.value("data").toObject().value("id").toString();
m_shot->m_url = object.value("data").toObject().value("link").toString();
m_shot->m_imgurDeleteHash = object.value("data").toObject().value("deletehash").toString();
- m_state = Task::State::Succeeded;
- finished = true;
- emit succeeded();
- return;
+ return Task::State::Succeeded;
}
-void ImgurUpload::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
+Net::NetRequest::Ptr ImgurUpload::make(ScreenShot::Ptr m_shot)
{
- setProgress(bytesReceived, bytesTotal);
- emit progress(bytesReceived, bytesTotal);
+ auto up = makeShared(m_shot->m_file);
+ up->m_url = std::move(BuildConfig.IMGUR_BASE_URL + "upload.json");
+ up->m_sink.reset(new Sink(m_shot));
+ return up;
}
diff --git a/launcher/screenshots/ImgurUpload.h b/launcher/screenshots/ImgurUpload.h
index 542255b2d..5867ad306 100644
--- a/launcher/screenshots/ImgurUpload.h
+++ b/launcher/screenshots/ImgurUpload.h
@@ -35,27 +35,36 @@
#pragma once
+#include
#include "Screenshot.h"
-#include "net/NetAction.h"
+#include "net/NetRequest.h"
-class ImgurUpload : public NetAction {
+class ImgurUpload : public Net::NetRequest {
public:
- using Ptr = shared_qobject_ptr;
+ class Sink : public Net::Sink {
+ public:
+ Sink(ScreenShot::Ptr shot) : m_shot(shot){};
+ virtual ~Sink() = default;
- explicit ImgurUpload(ScreenShot::Ptr shot);
- static Ptr make(ScreenShot::Ptr shot) { return Ptr(new ImgurUpload(shot)); }
- void init() override{};
+ public:
+ auto init(QNetworkRequest& request) -> Task::State override;
+ auto write(QByteArray& data) -> Task::State override;
+ auto abort() -> Task::State override;
+ auto finalize(QNetworkReply& reply) -> Task::State override;
+ auto hasLocalData() -> bool override { return false; }
- protected slots:
- void downloadProgress(qint64 bytesReceived, qint64 bytesTotal) override;
- void downloadError(QNetworkReply::NetworkError error) override;
- void downloadFinished() override;
- void downloadReadyRead() override {}
+ private:
+ ScreenShot::Ptr m_shot;
+ QByteArray m_output;
+ };
+ ImgurUpload(QFileInfo info) : m_fileInfo(info) {}
+ virtual ~ImgurUpload() = default;
- public slots:
- void executeTask() override;
+ static NetRequest::Ptr make(ScreenShot::Ptr m_shot);
+
+ void init() override;
private:
- ScreenShot::Ptr m_shot;
- bool finished = true;
+ virtual QNetworkReply* getReply(QNetworkRequest&) override;
+ const QFileInfo m_fileInfo;
};
diff --git a/launcher/settings/SettingsObject.h b/launcher/settings/SettingsObject.h
index 75631f247..f133f2f7f 100644
--- a/launcher/settings/SettingsObject.h
+++ b/launcher/settings/SettingsObject.h
@@ -26,8 +26,8 @@
class Setting;
class SettingsObject;
-typedef std::shared_ptr SettingsObjectPtr;
-typedef std::weak_ptr SettingsObjectWeakPtr;
+using SettingsObjectPtr = std::shared_ptr;
+using SettingsObjectWeakPtr = std::weak_ptr;
/*!
* \brief The SettingsObject handles communicating settings between the application and a
diff --git a/launcher/tasks/ConcurrentTask.h b/launcher/tasks/ConcurrentTask.h
index 8b696bf58..00b1d48d6 100644
--- a/launcher/tasks/ConcurrentTask.h
+++ b/launcher/tasks/ConcurrentTask.h
@@ -51,6 +51,9 @@ class ConcurrentTask : public Task {
explicit ConcurrentTask(QObject* parent = nullptr, QString task_name = "", int max_concurrent = 6);
~ConcurrentTask() override;
+ // safe to call before starting the task
+ void setMaxConcurrent(int max_concurrent) { m_total_max_size = max_concurrent; }
+
bool canAbort() const override { return true; }
inline auto isMultiStep() const -> bool override { return totalSize() > 1; }
diff --git a/launcher/tasks/Task.h b/launcher/tasks/Task.h
index 94b4089f3..883408c97 100644
--- a/launcher/tasks/Task.h
+++ b/launcher/tasks/Task.h
@@ -77,7 +77,7 @@ struct TaskStepProgress {
Q_DECLARE_METATYPE(TaskStepProgress)
-typedef QList> TaskStepProgressList;
+using TaskStepProgressList = QList>;
class Task : public QObject, public QRunnable {
Q_OBJECT
diff --git a/launcher/translations/TranslationsModel.cpp b/launcher/translations/TranslationsModel.cpp
index 933fe2d35..56ade8e32 100644
--- a/launcher/translations/TranslationsModel.cpp
+++ b/launcher/translations/TranslationsModel.cpp
@@ -206,7 +206,7 @@ void TranslationsModel::indexReceived()
reloadLocalFiles();
auto language = d->m_system_locale;
- if (!findLanguage(language)) {
+ if (!findLanguageAsOptional(language).has_value()) {
language = d->m_system_language;
}
selectLanguage(language);
@@ -417,14 +417,17 @@ int TranslationsModel::columnCount([[maybe_unused]] const QModelIndex& parent) c
return 2;
}
-Language* TranslationsModel::findLanguage(const QString& key)
+QVector::Iterator TranslationsModel::findLanguage(const QString& key)
{
- auto found = std::find_if(d->m_languages.begin(), d->m_languages.end(), [&](Language& lang) { return lang.key == key; });
- if (found == d->m_languages.end()) {
- return nullptr;
- } else {
- return found;
- }
+ return std::find_if(d->m_languages.begin(), d->m_languages.end(), [&](Language& lang) { return lang.key == key; });
+}
+
+std::optional TranslationsModel::findLanguageAsOptional(const QString& key)
+{
+ auto found = findLanguage(key);
+ if (found != d->m_languages.end())
+ return *found;
+ return {};
}
void TranslationsModel::setUseSystemLocale(bool useSystemLocale)
@@ -436,13 +439,13 @@ void TranslationsModel::setUseSystemLocale(bool useSystemLocale)
bool TranslationsModel::selectLanguage(QString key)
{
QString& langCode = key;
- auto langPtr = findLanguage(key);
+ auto langPtr = findLanguageAsOptional(key);
if (langCode.isEmpty()) {
d->no_language_set = true;
}
- if (!langPtr) {
+ if (!langPtr.has_value()) {
qWarning() << "Selected invalid language" << key << ", defaulting to" << defaultLangCode;
langCode = defaultLangCode;
} else {
@@ -527,9 +530,8 @@ bool TranslationsModel::selectLanguage(QString key)
QModelIndex TranslationsModel::selectedIndex()
{
auto found = findLanguage(d->m_selectedLanguage);
- if (found) {
- // QVector iterator freely converts to pointer to contained type
- return index(found - d->m_languages.begin(), 0, QModelIndex());
+ if (found != d->m_languages.end()) {
+ return index(std::distance(d->m_languages.begin(), found), 0, QModelIndex());
}
return QModelIndex();
}
@@ -562,8 +564,8 @@ void TranslationsModel::updateLanguage(QString key)
qWarning() << "Cannot update builtin language" << key;
return;
}
- auto found = findLanguage(key);
- if (!found) {
+ auto found = findLanguageAsOptional(key);
+ if (!found.has_value()) {
qWarning() << "Cannot update invalid language" << key;
return;
}
@@ -578,8 +580,8 @@ void TranslationsModel::downloadTranslation(QString key)
d->m_nextDownload = key;
return;
}
- auto lang = findLanguage(key);
- if (!lang) {
+ auto lang = findLanguageAsOptional(key);
+ if (!lang.has_value()) {
qWarning() << "Will not download an unknown translation" << key;
return;
}
diff --git a/launcher/translations/TranslationsModel.h b/launcher/translations/TranslationsModel.h
index cff23ce74..96a0e9f8b 100644
--- a/launcher/translations/TranslationsModel.h
+++ b/launcher/translations/TranslationsModel.h
@@ -17,6 +17,7 @@
#include