Compare commits
103 Commits
develop
...
release-7.
Author | SHA1 | Date | |
---|---|---|---|
|
770d5c92bc | ||
|
f326db11f1 | ||
|
f8c6a33134 | ||
|
64228bdddf | ||
|
88be40c4bf | ||
|
d6095358ad | ||
|
f7f2049223 | ||
|
cc43485650 | ||
|
b92d617df0 | ||
|
7b971a08a8 | ||
|
7cc3d34498 | ||
|
d9df60368c | ||
|
ba49afcc6a | ||
|
685badb8cf | ||
|
29fde6a322 | ||
|
e65d48bf36 | ||
|
38144b3661 | ||
|
3a0e30684e | ||
|
0fe184251d | ||
|
f51a66dad5 | ||
|
071e86fe68 | ||
|
d7a02ee456 | ||
|
648a69594f | ||
|
70fdfd1526 | ||
|
1e324e3e2f | ||
|
8bbe307a31 | ||
|
c946b2c4fb | ||
|
2dcfab0a19 | ||
|
b0b9b89bce | ||
|
862f4fb061 | ||
|
7c5d07e74d | ||
|
fe1ea7240e | ||
|
6f1d594f1c | ||
|
7baaf83f1d | ||
|
bab7105820 | ||
|
1389b74a8a | ||
|
5aa3aabdf9 | ||
|
42d2c7446a | ||
|
71a7358287 | ||
|
eaf125c31a | ||
|
f7e018d41a | ||
|
7d6e07ea71 | ||
|
cd011a097b | ||
|
055bcc2721 | ||
|
4bc4b29d5b | ||
|
82a9e7d372 | ||
|
cb81cadee3 | ||
|
0d8283df97 | ||
|
d4014534eb | ||
|
3c5ec5d967 | ||
|
e3625cad91 | ||
|
bb945c5165 | ||
|
daa5fcce67 | ||
|
2bcebe2989 | ||
|
27f6debdaf | ||
|
7926170073 | ||
|
79537f2948 | ||
|
2b3021b7c2 | ||
|
b11b86e026 | ||
|
a0ddd85b32 | ||
|
6f86e8b66e | ||
|
6cd259becd | ||
|
d9de326f22 | ||
|
1d4cf0fd03 | ||
|
a65e4af365 | ||
|
721ac015f3 | ||
|
c5572a5e0b | ||
|
81757717f7 | ||
|
1ab35357e9 | ||
|
7025f75903 | ||
|
3cc68fcea4 | ||
|
34be098f12 | ||
|
fb5655085f | ||
|
316ef9b725 | ||
|
5fdbc9d75e | ||
|
6464127d05 | ||
|
c349eff522 | ||
|
5fdf73a2ff | ||
|
6963bfc6c9 | ||
|
a25a5a7c9f | ||
|
20e9bf0e11 | ||
|
21ccf47ea7 | ||
|
6856c2f922 | ||
|
e52fd9d4fe | ||
|
05056e1abf | ||
|
9df3c5d3c0 | ||
|
1a4ea3b1cd | ||
|
0a7a7d9bfd | ||
|
8b017f9a5f | ||
|
b1fe4d1d93 | ||
|
80463f9761 | ||
|
567af5b22d | ||
|
7ed15b2687 | ||
|
3d502b12a9 | ||
|
f0d1df9139 | ||
|
0fcaa336dc | ||
|
949da6b50e | ||
|
fd0ba70080 | ||
|
89dd981dd7 | ||
|
30fda57c64 | ||
|
1d75c58d6c | ||
|
34d80a8c75 | ||
|
05360c1103 |
@ -1,17 +1,16 @@
|
|||||||
---
|
---
|
||||||
BasedOnStyle: Chromium
|
Language: Cpp
|
||||||
|
BasedOnStyle: Chromium
|
||||||
IndentWidth: 4
|
IndentWidth: 4
|
||||||
|
AlignConsecutiveMacros: false
|
||||||
|
AlignConsecutiveAssignments: false
|
||||||
AllowShortIfStatementsOnASingleLine: false
|
AllowShortIfStatementsOnASingleLine: false
|
||||||
ColumnLimit: 140
|
|
||||||
---
|
|
||||||
Language: Cpp
|
|
||||||
AlignConsecutiveMacros: None
|
|
||||||
AlignConsecutiveAssignments: None
|
|
||||||
BraceWrapping:
|
BraceWrapping:
|
||||||
AfterFunction: true
|
AfterFunction: true
|
||||||
SplitEmptyFunction: false
|
SplitEmptyFunction: false
|
||||||
SplitEmptyRecord: false
|
SplitEmptyRecord: false
|
||||||
SplitEmptyNamespace: false
|
SplitEmptyNamespace: false
|
||||||
BreakBeforeBraces: Custom
|
BreakBeforeBraces: Custom
|
||||||
BreakConstructorInitializers: BeforeComma
|
BreakConstructorInitializers: BeforeComma
|
||||||
|
ColumnLimit: 140
|
||||||
Cpp11BracedListStyle: false
|
Cpp11BracedListStyle: false
|
||||||
|
@ -1,4 +0,0 @@
|
|||||||
Checks:
|
|
||||||
- modernize-use-using
|
|
||||||
|
|
||||||
SystemHeaders: false
|
|
@ -1,8 +0,0 @@
|
|||||||
# EditorConfig specs and documentation: https://EditorConfig.org
|
|
||||||
|
|
||||||
# top-most EditorConfig file
|
|
||||||
root = true
|
|
||||||
|
|
||||||
# C++ Code Style settings
|
|
||||||
[*.{c++,cc,cpp,cppm,cxx,h,h++,hh,hpp,hxx,inl,ipp,ixx,tlh,tli}]
|
|
||||||
cpp_generate_documentation_comments = doxygen_slash_star
|
|
32
.github/workflows/backport.yml
vendored
32
.github/workflows/backport.yml
vendored
@ -1,32 +0,0 @@
|
|||||||
name: Backport
|
|
||||||
on:
|
|
||||||
pull_request_target:
|
|
||||||
types: [closed, labeled]
|
|
||||||
|
|
||||||
# WARNING:
|
|
||||||
# When extending this action, be aware that $GITHUB_TOKEN allows write access to
|
|
||||||
# the GitHub repository. This means that it should not evaluate user input in a
|
|
||||||
# way that allows code injection.
|
|
||||||
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
backport:
|
|
||||||
permissions:
|
|
||||||
contents: write # for korthout/backport-action to create branch
|
|
||||||
pull-requests: write # for korthout/backport-action to create PR to backport
|
|
||||||
name: Backport Pull Request
|
|
||||||
if: github.repository_owner == 'PrismLauncher' && github.event.pull_request.merged == true && (github.event_name != 'labeled' || startsWith('backport', github.event.label.name))
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
with:
|
|
||||||
ref: ${{ github.event.pull_request.head.sha }}
|
|
||||||
- name: Create backport PRs
|
|
||||||
uses: korthout/backport-action@v2.1.0
|
|
||||||
with:
|
|
||||||
# Config README: https://github.com/korthout/backport-action#backport-action
|
|
||||||
pull_description: |-
|
|
||||||
Bot-based backport to `${target_branch}`, triggered by a label in #${pull_number}.
|
|
||||||
|
|
176
.github/workflows/build.yml
vendored
176
.github/workflows/build.yml
vendored
@ -24,12 +24,6 @@ on:
|
|||||||
CACHIX_AUTH_TOKEN:
|
CACHIX_AUTH_TOKEN:
|
||||||
description: Private token for authenticating against Cachix cache
|
description: Private token for authenticating against Cachix cache
|
||||||
required: false
|
required: false
|
||||||
GPG_PRIVATE_KEY:
|
|
||||||
description: Private key for AppImage signing
|
|
||||||
required: false
|
|
||||||
GPG_PRIVATE_KEY_ID:
|
|
||||||
description: ID for the GPG_PRIVATE_KEY, to select the signing key
|
|
||||||
required: false
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
@ -37,43 +31,56 @@ jobs:
|
|||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
include:
|
include:
|
||||||
|
|
||||||
- os: ubuntu-20.04
|
- os: ubuntu-20.04
|
||||||
qt_ver: 5
|
qt_ver: 5
|
||||||
|
|
||||||
- os: ubuntu-20.04
|
- os: ubuntu-20.04
|
||||||
qt_ver: 6
|
qt_ver: 6
|
||||||
qt_host: linux
|
qt_host: linux
|
||||||
qt_arch: ""
|
qt_arch: ''
|
||||||
qt_version: "6.2.4"
|
qt_version: '6.2.4'
|
||||||
qt_modules: "qt5compat qtimageformats"
|
qt_modules: 'qt5compat qtimageformats'
|
||||||
qt_tools: ""
|
qt_tools: ''
|
||||||
|
|
||||||
- os: windows-2022
|
- os: windows-2022
|
||||||
name: "Windows-MinGW-w64"
|
name: "Windows-MinGW-w64"
|
||||||
msystem: clang64
|
msystem: clang64
|
||||||
vcvars_arch: "amd64_x86"
|
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'
|
||||||
|
|
||||||
- os: windows-2022
|
- os: windows-2022
|
||||||
name: "Windows-MSVC"
|
name: "Windows-MSVC"
|
||||||
msystem: ""
|
msystem: ''
|
||||||
architecture: "x64"
|
architecture: 'x64'
|
||||||
vcvars_arch: "amd64"
|
vcvars_arch: 'amd64'
|
||||||
qt_ver: 6
|
qt_ver: 6
|
||||||
qt_host: windows
|
qt_host: windows
|
||||||
qt_arch: ''
|
qt_arch: ''
|
||||||
qt_version: '6.6.0'
|
qt_version: '6.5.1'
|
||||||
qt_modules: 'qt5compat qtimageformats'
|
qt_modules: 'qt5compat qtimageformats'
|
||||||
qt_tools: ''
|
qt_tools: ''
|
||||||
|
|
||||||
- os: windows-2022
|
- os: windows-2022
|
||||||
name: "Windows-MSVC-arm64"
|
name: "Windows-MSVC-arm64"
|
||||||
msystem: ""
|
msystem: ''
|
||||||
architecture: "arm64"
|
architecture: 'arm64'
|
||||||
vcvars_arch: "amd64_arm64"
|
vcvars_arch: 'amd64_arm64'
|
||||||
qt_ver: 6
|
qt_ver: 6
|
||||||
qt_host: windows
|
qt_host: windows
|
||||||
qt_arch: 'win64_msvc2019_arm64'
|
qt_arch: 'win64_msvc2019_arm64'
|
||||||
qt_version: '6.6.0'
|
qt_version: '6.5.1'
|
||||||
qt_modules: 'qt5compat qtimageformats'
|
qt_modules: 'qt5compat qtimageformats'
|
||||||
qt_tools: ''
|
qt_tools: ''
|
||||||
|
|
||||||
@ -83,7 +90,7 @@ jobs:
|
|||||||
qt_ver: 6
|
qt_ver: 6
|
||||||
qt_host: mac
|
qt_host: mac
|
||||||
qt_arch: ''
|
qt_arch: ''
|
||||||
qt_version: '6.6.0'
|
qt_version: '6.5.0'
|
||||||
qt_modules: 'qt5compat qtimageformats'
|
qt_modules: 'qt5compat qtimageformats'
|
||||||
qt_tools: ''
|
qt_tools: ''
|
||||||
|
|
||||||
@ -92,9 +99,9 @@ jobs:
|
|||||||
macosx_deployment_target: 10.13
|
macosx_deployment_target: 10.13
|
||||||
qt_ver: 5
|
qt_ver: 5
|
||||||
qt_host: mac
|
qt_host: mac
|
||||||
qt_version: "5.15.2"
|
qt_version: '5.15.2'
|
||||||
qt_modules: ""
|
qt_modules: ''
|
||||||
qt_tools: ""
|
qt_tools: ''
|
||||||
|
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
|
|
||||||
@ -112,11 +119,11 @@ jobs:
|
|||||||
# PREPARE
|
# PREPARE
|
||||||
##
|
##
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
submodules: "true"
|
submodules: 'true'
|
||||||
|
|
||||||
- name: "Setup MSYS2"
|
- name: 'Setup MSYS2'
|
||||||
if: runner.os == 'Windows' && matrix.msystem != ''
|
if: runner.os == 'Windows' && matrix.msystem != ''
|
||||||
uses: msys2/setup-msys2@v2
|
uses: msys2/setup-msys2@v2
|
||||||
with:
|
with:
|
||||||
@ -145,18 +152,18 @@ jobs:
|
|||||||
|
|
||||||
- name: Setup ccache
|
- name: Setup ccache
|
||||||
if: (runner.os != 'Windows' || matrix.msystem == '') && inputs.build_type == 'Debug'
|
if: (runner.os != 'Windows' || matrix.msystem == '') && inputs.build_type == 'Debug'
|
||||||
uses: hendrikmuhs/ccache-action@v1.2.10
|
uses: hendrikmuhs/ccache-action@v1.2.9
|
||||||
with:
|
with:
|
||||||
key: ${{ matrix.os }}-qt${{ matrix.qt_ver }}-${{ matrix.architecture }}
|
key: ${{ matrix.os }}-qt${{ matrix.qt_ver }}-${{ matrix.architecture }}
|
||||||
|
|
||||||
- name: Retrieve ccache cache (Windows MinGW-w64)
|
- name: Retrieve ccache cache (Windows MinGW-w64)
|
||||||
if: runner.os == 'Windows' && matrix.msystem != '' && inputs.build_type == 'Debug'
|
if: runner.os == 'Windows' && matrix.msystem != '' && inputs.build_type == 'Debug'
|
||||||
uses: actions/cache@v3.3.2
|
uses: actions/cache@v3.3.1
|
||||||
with:
|
with:
|
||||||
path: '${{ github.workspace }}\.ccache'
|
path: '${{ github.workspace }}\.ccache'
|
||||||
key: ${{ matrix.os }}-mingw-w64-ccache-${{ github.run_id }}
|
key: ${{ matrix.os }}-mingw-w64-ccache-${{ github.run_id }}
|
||||||
restore-keys: |
|
restore-keys: |
|
||||||
${{ matrix.os }}-mingw-w64-ccache
|
${{ matrix.os }}-mingw-w64-ccache
|
||||||
|
|
||||||
- name: Setup ccache (Windows MinGW-w64)
|
- name: Setup ccache (Windows MinGW-w64)
|
||||||
if: runner.os == 'Windows' && matrix.msystem != '' && inputs.build_type == 'Debug'
|
if: runner.os == 'Windows' && matrix.msystem != '' && inputs.build_type == 'Debug'
|
||||||
@ -201,35 +208,35 @@ jobs:
|
|||||||
if: runner.os == 'Windows' && matrix.architecture == 'arm64'
|
if: runner.os == 'Windows' && matrix.architecture == 'arm64'
|
||||||
uses: jurplel/install-qt-action@v3
|
uses: jurplel/install-qt-action@v3
|
||||||
with:
|
with:
|
||||||
aqtversion: "==3.1.*"
|
aqtversion: '==3.1.*'
|
||||||
py7zrversion: ">=0.20.2"
|
py7zrversion: '>=0.20.2'
|
||||||
version: ${{ matrix.qt_version }}
|
version: ${{ matrix.qt_version }}
|
||||||
host: "windows"
|
host: 'windows'
|
||||||
target: "desktop"
|
target: 'desktop'
|
||||||
arch: ""
|
arch: ''
|
||||||
modules: ${{ matrix.qt_modules }}
|
modules: ${{ matrix.qt_modules }}
|
||||||
tools: ${{ matrix.qt_tools }}
|
tools: ${{ matrix.qt_tools }}
|
||||||
cache: ${{ inputs.is_qt_cached }}
|
cache: ${{ inputs.is_qt_cached }}
|
||||||
cache-key-prefix: host-qt-arm64-windows
|
cache-key-prefix: host-qt-arm64-windows
|
||||||
dir: ${{ github.workspace }}\HostQt
|
dir: ${{ github.workspace }}\HostQt
|
||||||
set-env: false
|
set-env: false
|
||||||
|
|
||||||
- name: Install Qt (macOS, Linux, Qt 6 & Windows MSVC)
|
- 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 == '')
|
if: runner.os == 'Linux' && matrix.qt_ver == 6 || runner.os == 'macOS' || (runner.os == 'Windows' && matrix.msystem == '')
|
||||||
uses: jurplel/install-qt-action@v3
|
uses: jurplel/install-qt-action@v3
|
||||||
with:
|
with:
|
||||||
aqtversion: "==3.1.*"
|
aqtversion: '==3.1.*'
|
||||||
py7zrversion: ">=0.20.2"
|
py7zrversion: '>=0.20.2'
|
||||||
version: ${{ matrix.qt_version }}
|
version: ${{ matrix.qt_version }}
|
||||||
host: ${{ matrix.qt_host }}
|
host: ${{ matrix.qt_host }}
|
||||||
target: "desktop"
|
target: 'desktop'
|
||||||
arch: ${{ matrix.qt_arch }}
|
arch: ${{ matrix.qt_arch }}
|
||||||
modules: ${{ matrix.qt_modules }}
|
modules: ${{ matrix.qt_modules }}
|
||||||
tools: ${{ matrix.qt_tools }}
|
tools: ${{ matrix.qt_tools }}
|
||||||
cache: ${{ inputs.is_qt_cached }}
|
cache: ${{ inputs.is_qt_cached }}
|
||||||
|
|
||||||
- name: Install MSVC (Windows MSVC)
|
- 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
|
uses: ilammy/msvc-dev-cmd@v1
|
||||||
with:
|
with:
|
||||||
vsversion: 2022
|
vsversion: 2022
|
||||||
@ -242,8 +249,6 @@ jobs:
|
|||||||
wget "https://github.com/linuxdeploy/linuxdeploy-plugin-appimage/releases/download/continuous/linuxdeploy-plugin-appimage-x86_64.AppImage"
|
wget "https://github.com/linuxdeploy/linuxdeploy-plugin-appimage/releases/download/continuous/linuxdeploy-plugin-appimage-x86_64.AppImage"
|
||||||
wget "https://github.com/linuxdeploy/linuxdeploy-plugin-qt/releases/download/continuous/linuxdeploy-plugin-qt-x86_64.AppImage"
|
wget "https://github.com/linuxdeploy/linuxdeploy-plugin-qt/releases/download/continuous/linuxdeploy-plugin-qt-x86_64.AppImage"
|
||||||
|
|
||||||
wget "https://github.com/AppImageCommunity/AppImageUpdate/releases/download/continuous/AppImageUpdate-x86_64.AppImage"
|
|
||||||
|
|
||||||
${{ github.workspace }}/.github/scripts/prepare_JREs.sh
|
${{ github.workspace }}/.github/scripts/prepare_JREs.sh
|
||||||
sudo apt install libopengl0
|
sudo apt install libopengl0
|
||||||
|
|
||||||
@ -270,12 +275,12 @@ jobs:
|
|||||||
if: runner.os == 'Windows' && matrix.msystem != ''
|
if: runner.os == 'Windows' && matrix.msystem != ''
|
||||||
shell: msys2 {0}
|
shell: msys2 {0}
|
||||||
run: |
|
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 -DLauncher_BUILD_ARTIFACT=${{ matrix.name }}-Qt${{ matrix.qt_ver }} -G Ninja
|
cmake -S . -B ${{ env.BUILD_DIR }} -DCMAKE_INSTALL_PREFIX=${{ env.INSTALL_DIR }} -DCMAKE_BUILD_TYPE=${{ inputs.build_type }} -DENABLE_LTO=ON -DLauncher_BUILD_PLATFORM=official -DCMAKE_C_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DCMAKE_CXX_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DLauncher_QT_VERSION_MAJOR=6 -DCMAKE_OBJDUMP=/mingw64/bin/objdump.exe -G Ninja
|
||||||
|
|
||||||
- name: Configure CMake (Windows MSVC)
|
- name: Configure CMake (Windows MSVC)
|
||||||
if: runner.os == 'Windows' && matrix.msystem == ''
|
if: runner.os == 'Windows' && matrix.msystem == ''
|
||||||
run: |
|
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 -DLauncher_BUILD_ARTIFACT=${{ matrix.name }}-Qt${{ matrix.qt_ver }}
|
cmake -S . -B ${{ env.BUILD_DIR }} -DCMAKE_INSTALL_PREFIX=${{ env.INSTALL_DIR }} -DCMAKE_BUILD_TYPE=${{ inputs.build_type }} -DENABLE_LTO=ON -DLauncher_BUILD_PLATFORM=official -DLauncher_QT_VERSION_MAJOR=${{ matrix.qt_ver }} -DCMAKE_MSVC_RUNTIME_LIBRARY="MultiThreadedDLL" -A${{ matrix.architecture}} -DLauncher_FORCE_BUNDLED_LIBS=ON
|
||||||
# https://github.com/ccache/ccache/wiki/MS-Visual-Studio (I coudn't figure out the compiler prefix)
|
# https://github.com/ccache/ccache/wiki/MS-Visual-Studio (I coudn't figure out the compiler prefix)
|
||||||
if ("${{ env.CCACHE_VAR }}")
|
if ("${{ env.CCACHE_VAR }}")
|
||||||
{
|
{
|
||||||
@ -290,7 +295,7 @@ jobs:
|
|||||||
- name: Configure CMake (Linux)
|
- name: Configure CMake (Linux)
|
||||||
if: runner.os == 'Linux'
|
if: runner.os == 'Linux'
|
||||||
run: |
|
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 }} -DLauncher_BUILD_ARTIFACT=Linux-Qt${{ matrix.qt_ver }} -G Ninja
|
cmake -S . -B ${{ env.BUILD_DIR }} -DCMAKE_INSTALL_PREFIX=/usr -DCMAKE_BUILD_TYPE=${{ inputs.build_type }} -DENABLE_LTO=ON -DLauncher_BUILD_PLATFORM=official -DCMAKE_C_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DCMAKE_CXX_COMPILER_LAUNCHER=${{ env.CCACHE_VAR }} -DLauncher_QT_VERSION_MAJOR=${{ matrix.qt_ver }} -G Ninja
|
||||||
|
|
||||||
##
|
##
|
||||||
# BUILD
|
# BUILD
|
||||||
@ -330,7 +335,7 @@ jobs:
|
|||||||
- name: Test (Windows MSVC)
|
- name: Test (Windows MSVC)
|
||||||
if: runner.os == 'Windows' && matrix.msystem == '' && matrix.architecture != 'arm64'
|
if: runner.os == 'Windows' && matrix.msystem == '' && matrix.architecture != 'arm64'
|
||||||
run: |
|
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
|
# PACKAGE BUILDS
|
||||||
@ -372,7 +377,7 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
cmake --install ${{ env.BUILD_DIR }}
|
cmake --install ${{ env.BUILD_DIR }}
|
||||||
touch ${{ env.INSTALL_DIR }}/manifest.txt
|
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)
|
- name: Package (Windows MSVC)
|
||||||
if: runner.os == 'Windows' && matrix.msystem == ''
|
if: runner.os == 'Windows' && matrix.msystem == ''
|
||||||
@ -382,16 +387,17 @@ jobs:
|
|||||||
cd ${{ env.INSTALL_DIR }}
|
cd ${{ env.INSTALL_DIR }}
|
||||||
if ("${{ matrix.qt_ver }}" -eq "5")
|
if ("${{ matrix.qt_ver }}" -eq "5")
|
||||||
{
|
{
|
||||||
Copy-Item ${{ runner.workspace }}/Qt/Tools/OpenSSL/Win_x86/bin/libcrypto-1_1.dll -Destination libcrypto-1_1.dll
|
Copy-Item D:/a/PrismLauncher/Qt/Tools/OpenSSL/Win_x86/bin/libcrypto-1_1.dll -Destination libcrypto-1_1.dll
|
||||||
Copy-Item ${{ runner.workspace }}/Qt/Tools/OpenSSL/Win_x86/bin/libssl-1_1.dll -Destination libssl-1_1.dll
|
Copy-Item D:/a/PrismLauncher/Qt/Tools/OpenSSL/Win_x86/bin/libssl-1_1.dll -Destination libssl-1_1.dll
|
||||||
}
|
}
|
||||||
cd ${{ github.workspace }}
|
cd ${{ github.workspace }}
|
||||||
|
|
||||||
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
|
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)
|
- name: Fetch codesign certificate (Windows)
|
||||||
if: runner.os == '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: |
|
run: |
|
||||||
echo '${{ secrets.WINDOWS_CODESIGN_CERT }}' | base64 --decode > codesign.pfx
|
echo '${{ secrets.WINDOWS_CODESIGN_CERT }}' | base64 --decode > codesign.pfx
|
||||||
|
|
||||||
@ -401,7 +407,7 @@ jobs:
|
|||||||
if (Get-Content ./codesign.pfx){
|
if (Get-Content ./codesign.pfx){
|
||||||
cd ${{ env.INSTALL_DIR }}
|
cd ${{ env.INSTALL_DIR }}
|
||||||
# We ship the exact same executable for portable and non-portable editions, so signing just once is fine
|
# 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 prismlauncher.exe prismlauncher_updater.exe prismlauncher_filelink.exe
|
SignTool sign /fd sha256 /td sha256 /f ../codesign.pfx /p '${{ secrets.WINDOWS_CODESIGN_PASSWORD }}' /tr http://timestamp.digicert.com prismlauncher.exe prismlauncher_filelink.exe
|
||||||
} else {
|
} else {
|
||||||
":warning: Skipped code signing for Windows, as certificate was not present." >> $env:GITHUB_STEP_SUMMARY
|
":warning: Skipped code signing for Windows, as certificate was not present." >> $env:GITHUB_STEP_SUMMARY
|
||||||
}
|
}
|
||||||
@ -419,7 +425,7 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
cp -r ${{ env.INSTALL_DIR }} ${{ env.INSTALL_PORTABLE_DIR }} # cmake install on Windows is slow, let's just copy instead
|
cp -r ${{ env.INSTALL_DIR }} ${{ env.INSTALL_PORTABLE_DIR }} # cmake install on Windows is slow, let's just copy instead
|
||||||
cmake --install ${{ env.BUILD_DIR }} --prefix ${{ env.INSTALL_PORTABLE_DIR }} --component portable
|
cmake --install ${{ env.BUILD_DIR }} --prefix ${{ env.INSTALL_PORTABLE_DIR }} --component portable
|
||||||
|
|
||||||
Get-ChildItem ${{ env.INSTALL_PORTABLE_DIR }} -Recurse | ForEach FullName | Resolve-Path -Relative | %{ $_.TrimStart('.\') } | %{ $_.TrimStart('${{ env.INSTALL_PORTABLE_DIR }}') } | %{ $_.TrimStart('\') } | Out-File -FilePath ${{ env.INSTALL_DIR }}/manifest.txt
|
Get-ChildItem ${{ env.INSTALL_PORTABLE_DIR }} -Recurse | ForEach FullName | Resolve-Path -Relative | %{ $_.TrimStart('.\') } | %{ $_.TrimStart('${{ env.INSTALL_PORTABLE_DIR }}') } | %{ $_.TrimStart('\') } | Out-File -FilePath ${{ env.INSTALL_DIR }}/manifest.txt
|
||||||
|
|
||||||
- name: Package (Windows, installer)
|
- name: Package (Windows, installer)
|
||||||
@ -460,15 +466,11 @@ jobs:
|
|||||||
- name: Package AppImage (Linux)
|
- name: Package AppImage (Linux)
|
||||||
if: runner.os == 'Linux' && matrix.qt_ver != 5
|
if: runner.os == 'Linux' && matrix.qt_ver != 5
|
||||||
shell: bash
|
shell: bash
|
||||||
env:
|
|
||||||
GPG_PRIVATE_KEY: ${{ secrets.GPG_PRIVATE_KEY }}
|
|
||||||
run: |
|
run: |
|
||||||
cmake --install ${{ env.BUILD_DIR }} --prefix ${{ env.INSTALL_APPIMAGE_DIR }}/usr
|
cmake --install ${{ env.BUILD_DIR }} --prefix ${{ env.INSTALL_APPIMAGE_DIR }}/usr
|
||||||
|
|
||||||
mv ${{ env.INSTALL_APPIMAGE_DIR }}/usr/share/metainfo/org.prismlauncher.PrismLauncher.metainfo.xml ${{ env.INSTALL_APPIMAGE_DIR }}/usr/share/metainfo/org.prismlauncher.PrismLauncher.appdata.xml
|
mv ${{ env.INSTALL_APPIMAGE_DIR }}/usr/share/metainfo/org.prismlauncher.PrismLauncher.metainfo.xml ${{ env.INSTALL_APPIMAGE_DIR }}/usr/share/metainfo/org.prismlauncher.PrismLauncher.appdata.xml
|
||||||
export "NO_APPSTREAM=1" # we have to skip appstream checking because appstream on ubuntu 20.04 is outdated
|
export "NO_APPSTREAM=1" # we have to skip appstream checking because appstream on ubuntu 20.04 is outdated
|
||||||
|
export OUTPUT="PrismLauncher-${{ runner.os }}-${{ env.VERSION }}-${{ inputs.build_type }}-x86_64.AppImage"
|
||||||
export OUTPUT="PrismLauncher-Linux-x86_64.AppImage"
|
|
||||||
|
|
||||||
chmod +x linuxdeploy-*.AppImage
|
chmod +x linuxdeploy-*.AppImage
|
||||||
|
|
||||||
@ -479,8 +481,8 @@ jobs:
|
|||||||
|
|
||||||
cp -r ${{ github.workspace }}/JREs/jre17/* ${{ env.INSTALL_APPIMAGE_DIR }}/usr/lib/jvm/java-17-openjdk
|
cp -r ${{ github.workspace }}/JREs/jre17/* ${{ env.INSTALL_APPIMAGE_DIR }}/usr/lib/jvm/java-17-openjdk
|
||||||
|
|
||||||
cp -r ${{ runner.workspace }}/Qt/${{ matrix.qt_version }}/gcc_64/plugins/iconengines/* ${{ env.INSTALL_APPIMAGE_DIR }}/usr/plugins/iconengines
|
cp -r /home/runner/work/PrismLauncher/Qt/${{ matrix.qt_version }}/gcc_64/plugins/iconengines/* ${{ env.INSTALL_APPIMAGE_DIR }}/usr/plugins/iconengines
|
||||||
|
|
||||||
cp /usr/lib/x86_64-linux-gnu/libcrypto.so.1.1 ${{ env.INSTALL_APPIMAGE_DIR }}/usr/lib/
|
cp /usr/lib/x86_64-linux-gnu/libcrypto.so.1.1 ${{ env.INSTALL_APPIMAGE_DIR }}/usr/lib/
|
||||||
cp /usr/lib/x86_64-linux-gnu/libssl.so.1.1 ${{ env.INSTALL_APPIMAGE_DIR }}/usr/lib/
|
cp /usr/lib/x86_64-linux-gnu/libssl.so.1.1 ${{ env.INSTALL_APPIMAGE_DIR }}/usr/lib/
|
||||||
cp /usr/lib/x86_64-linux-gnu/libOpenGL.so.0* ${{ env.INSTALL_APPIMAGE_DIR }}/usr/lib/
|
cp /usr/lib/x86_64-linux-gnu/libOpenGL.so.0* ${{ env.INSTALL_APPIMAGE_DIR }}/usr/lib/
|
||||||
@ -492,25 +494,8 @@ jobs:
|
|||||||
LD_LIBRARY_PATH="${LD_LIBRARY_PATH}:${{ env.INSTALL_APPIMAGE_DIR }}/usr/lib/jvm/java-17-openjdk/lib"
|
LD_LIBRARY_PATH="${LD_LIBRARY_PATH}:${{ env.INSTALL_APPIMAGE_DIR }}/usr/lib/jvm/java-17-openjdk/lib"
|
||||||
export LD_LIBRARY_PATH
|
export LD_LIBRARY_PATH
|
||||||
|
|
||||||
chmod +x AppImageUpdate-x86_64.AppImage
|
|
||||||
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|PrismLauncher-Linux-x86_64.AppImage.zsync"
|
|
||||||
|
|
||||||
if [ '${{ secrets.GPG_PRIVATE_KEY_ID }}' != '' ]; then
|
|
||||||
export SIGN=1
|
|
||||||
export SIGN_KEY=${{ secrets.GPG_PRIVATE_KEY_ID }}
|
|
||||||
mkdir -p ~/.gnupg/
|
|
||||||
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
|
|
||||||
fi
|
|
||||||
|
|
||||||
./linuxdeploy-x86_64.AppImage --appdir ${{ env.INSTALL_APPIMAGE_DIR }} --output appimage --plugin qt -i ${{ env.INSTALL_APPIMAGE_DIR }}/usr/share/icons/hicolor/scalable/apps/org.prismlauncher.PrismLauncher.svg
|
./linuxdeploy-x86_64.AppImage --appdir ${{ env.INSTALL_APPIMAGE_DIR }} --output appimage --plugin qt -i ${{ env.INSTALL_APPIMAGE_DIR }}/usr/share/icons/hicolor/scalable/apps/org.prismlauncher.PrismLauncher.svg
|
||||||
|
|
||||||
mv "PrismLauncher-Linux-x86_64.AppImage" "PrismLauncher-Linux-${{ env.VERSION }}-${{ inputs.build_type }}-x86_64.AppImage"
|
|
||||||
|
|
||||||
##
|
##
|
||||||
# UPLOAD BUILDS
|
# UPLOAD BUILDS
|
||||||
##
|
##
|
||||||
@ -547,14 +532,14 @@ jobs:
|
|||||||
if: runner.os == 'Linux' && matrix.qt_ver != 6
|
if: runner.os == 'Linux' && matrix.qt_ver != 6
|
||||||
uses: actions/upload-artifact@v3
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: PrismLauncher-${{ runner.os }}-Qt5-${{ env.VERSION }}-${{ inputs.build_type }}
|
name: PrismLauncher-${{ runner.os }}-${{ env.VERSION }}-${{ inputs.build_type }}
|
||||||
path: PrismLauncher.tar.gz
|
path: PrismLauncher.tar.gz
|
||||||
|
|
||||||
- name: Upload binary tarball (Linux, portable, Qt 5)
|
- name: Upload binary tarball (Linux, portable, Qt 5)
|
||||||
if: runner.os == 'Linux' && matrix.qt_ver != 6
|
if: runner.os == 'Linux' && matrix.qt_ver != 6
|
||||||
uses: actions/upload-artifact@v3
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: PrismLauncher-${{ runner.os }}-Qt5-Portable-${{ env.VERSION }}-${{ inputs.build_type }}
|
name: PrismLauncher-${{ runner.os }}-Portable-${{ env.VERSION }}-${{ inputs.build_type }}
|
||||||
path: PrismLauncher-portable.tar.gz
|
path: PrismLauncher-portable.tar.gz
|
||||||
|
|
||||||
- name: Upload binary tarball (Linux, Qt 6)
|
- name: Upload binary tarball (Linux, Qt 6)
|
||||||
@ -578,13 +563,6 @@ jobs:
|
|||||||
name: PrismLauncher-${{ runner.os }}-${{ env.VERSION }}-${{ inputs.build_type }}-x86_64.AppImage
|
name: PrismLauncher-${{ runner.os }}-${{ env.VERSION }}-${{ inputs.build_type }}-x86_64.AppImage
|
||||||
path: PrismLauncher-${{ runner.os }}-${{ env.VERSION }}-${{ inputs.build_type }}-x86_64.AppImage
|
path: PrismLauncher-${{ runner.os }}-${{ env.VERSION }}-${{ inputs.build_type }}-x86_64.AppImage
|
||||||
|
|
||||||
- name: Upload AppImage Zsync (Linux)
|
|
||||||
if: runner.os == 'Linux' && matrix.qt_ver != 5
|
|
||||||
uses: actions/upload-artifact@v3
|
|
||||||
with:
|
|
||||||
name: PrismLauncher-${{ runner.os }}-${{ env.VERSION }}-${{ inputs.build_type }}-x86_64.AppImage.zsync
|
|
||||||
path: PrismLauncher-Linux-x86_64.AppImage.zsync
|
|
||||||
|
|
||||||
- name: ccache stats (Windows MinGW-w64)
|
- name: ccache stats (Windows MinGW-w64)
|
||||||
if: runner.os == 'Windows' && matrix.msystem != ''
|
if: runner.os == 'Windows' && matrix.msystem != ''
|
||||||
shell: msys2 {0}
|
shell: msys2 {0}
|
||||||
@ -598,13 +576,13 @@ jobs:
|
|||||||
options: --privileged
|
options: --privileged
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v3
|
||||||
if: inputs.build_type == 'Debug'
|
if: inputs.build_type == 'Debug'
|
||||||
with:
|
with:
|
||||||
submodules: "true"
|
submodules: 'true'
|
||||||
- name: Build Flatpak (Linux)
|
- name: Build Flatpak (Linux)
|
||||||
if: inputs.build_type == 'Debug'
|
if: inputs.build_type == 'Debug'
|
||||||
uses: flatpak/flatpak-github-actions/flatpak-builder@v6
|
uses: flatpak/flatpak-github-actions/flatpak-builder@v6
|
||||||
with:
|
with:
|
||||||
bundle: "Prism Launcher.flatpak"
|
bundle: "Prism Launcher.flatpak"
|
||||||
manifest-path: flatpak/org.prismlauncher.PrismLauncher.yml
|
manifest-path: flatpak/org.prismlauncher.PrismLauncher.yml
|
||||||
|
2
.github/workflows/codeql.yml
vendored
2
.github/workflows/codeql.yml
vendored
@ -8,7 +8,7 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
submodules: 'true'
|
submodules: 'true'
|
||||||
|
|
||||||
|
29
.github/workflows/trigger_builds.yml
vendored
29
.github/workflows/trigger_builds.yml
vendored
@ -3,25 +3,26 @@ name: Build Application
|
|||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches-ignore:
|
branches-ignore:
|
||||||
- "renovate/**"
|
- 'renovate/**'
|
||||||
paths-ignore:
|
paths-ignore:
|
||||||
- "**.md"
|
- '**.md'
|
||||||
- "**/LICENSE"
|
- '**/LICENSE'
|
||||||
- "flake.lock"
|
- 'flake.lock'
|
||||||
- "packages/**"
|
- 'packages/**'
|
||||||
- ".github/ISSUE_TEMPLATE/**"
|
- '.github/ISSUE_TEMPLATE/**'
|
||||||
- ".markdownlint**"
|
- '.markdownlint**'
|
||||||
pull_request:
|
pull_request:
|
||||||
paths-ignore:
|
paths-ignore:
|
||||||
- "**.md"
|
- '**.md'
|
||||||
- "**/LICENSE"
|
- '**/LICENSE'
|
||||||
- "flake.lock"
|
- 'flake.lock'
|
||||||
- "packages/**"
|
- 'packages/**'
|
||||||
- ".github/ISSUE_TEMPLATE/**"
|
- '.github/ISSUE_TEMPLATE/**'
|
||||||
- ".markdownlint**"
|
- '.markdownlint**'
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
|
||||||
build_debug:
|
build_debug:
|
||||||
name: Build Debug
|
name: Build Debug
|
||||||
uses: ./.github/workflows/build.yml
|
uses: ./.github/workflows/build.yml
|
||||||
@ -33,5 +34,3 @@ jobs:
|
|||||||
WINDOWS_CODESIGN_CERT: ${{ secrets.WINDOWS_CODESIGN_CERT }}
|
WINDOWS_CODESIGN_CERT: ${{ secrets.WINDOWS_CODESIGN_CERT }}
|
||||||
WINDOWS_CODESIGN_PASSWORD: ${{ secrets.WINDOWS_CODESIGN_PASSWORD }}
|
WINDOWS_CODESIGN_PASSWORD: ${{ secrets.WINDOWS_CODESIGN_PASSWORD }}
|
||||||
CACHIX_AUTH_TOKEN: ${{ secrets.CACHIX_AUTH_TOKEN }}
|
CACHIX_AUTH_TOKEN: ${{ secrets.CACHIX_AUTH_TOKEN }}
|
||||||
GPG_PRIVATE_KEY: ${{ secrets.GPG_PRIVATE_KEY }}
|
|
||||||
GPG_PRIVATE_KEY_ID: ${{ secrets.GPG_PRIVATE_KEY_ID }}
|
|
||||||
|
33
.github/workflows/trigger_release.yml
vendored
33
.github/workflows/trigger_release.yml
vendored
@ -3,9 +3,10 @@ name: Build Application and Make Release
|
|||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
tags:
|
tags:
|
||||||
- "*"
|
- '*'
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
|
||||||
build_release:
|
build_release:
|
||||||
name: Build Release
|
name: Build Release
|
||||||
uses: ./.github/workflows/build.yml
|
uses: ./.github/workflows/build.yml
|
||||||
@ -17,8 +18,6 @@ jobs:
|
|||||||
WINDOWS_CODESIGN_CERT: ${{ secrets.WINDOWS_CODESIGN_CERT }}
|
WINDOWS_CODESIGN_CERT: ${{ secrets.WINDOWS_CODESIGN_CERT }}
|
||||||
WINDOWS_CODESIGN_PASSWORD: ${{ secrets.WINDOWS_CODESIGN_PASSWORD }}
|
WINDOWS_CODESIGN_PASSWORD: ${{ secrets.WINDOWS_CODESIGN_PASSWORD }}
|
||||||
CACHIX_AUTH_TOKEN: ${{ secrets.CACHIX_AUTH_TOKEN }}
|
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:
|
create_release:
|
||||||
needs: build_release
|
needs: build_release
|
||||||
@ -27,10 +26,10 @@ jobs:
|
|||||||
upload_url: ${{ steps.create_release.outputs.upload_url }}
|
upload_url: ${{ steps.create_release.outputs.upload_url }}
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
submodules: "true"
|
submodules: 'true'
|
||||||
path: "PrismLauncher-source"
|
path: 'PrismLauncher-source'
|
||||||
- name: Download artifacts
|
- name: Download artifacts
|
||||||
uses: actions/download-artifact@v3
|
uses: actions/download-artifact@v3
|
||||||
- name: Grab and store version
|
- name: Grab and store version
|
||||||
@ -41,11 +40,10 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
mv ${{ github.workspace }}/PrismLauncher-source PrismLauncher-${{ env.VERSION }}
|
mv ${{ github.workspace }}/PrismLauncher-source PrismLauncher-${{ env.VERSION }}
|
||||||
mv PrismLauncher-Linux-Qt6-Portable*/PrismLauncher-portable.tar.gz PrismLauncher-Linux-Qt6-Portable-${{ env.VERSION }}.tar.gz
|
mv PrismLauncher-Linux-Qt6-Portable*/PrismLauncher-portable.tar.gz PrismLauncher-Linux-Qt6-Portable-${{ env.VERSION }}.tar.gz
|
||||||
mv PrismLauncher-Linux-Qt6*/PrismLauncher.tar.gz PrismLauncher-Linux-Qt6-${{ env.VERSION }}.tar.gz
|
mv PrismLauncher-Linux-Qt6*/PrismLauncher.tar.gz PrismLauncher-Linux-Qt6-${{ env.VERSION }}.tar.gz
|
||||||
mv PrismLauncher-Linux-Qt5-Portable*/PrismLauncher-portable.tar.gz PrismLauncher-Linux-Qt5-Portable-${{ env.VERSION }}.tar.gz
|
mv PrismLauncher-Linux-Portable*/PrismLauncher-portable.tar.gz PrismLauncher-Linux-Portable-${{ env.VERSION }}.tar.gz
|
||||||
mv PrismLauncher-Linux-Qt5*/PrismLauncher.tar.gz PrismLauncher-Linux-Qt5-${{ env.VERSION }}.tar.gz
|
mv PrismLauncher-Linux*/PrismLauncher.tar.gz PrismLauncher-Linux-${{ env.VERSION }}.tar.gz
|
||||||
mv PrismLauncher-*.AppImage/PrismLauncher-*.AppImage PrismLauncher-Linux-x86_64.AppImage
|
mv PrismLauncher-*.AppImage/PrismLauncher-*.AppImage PrismLauncher-Linux-${{ env.VERSION }}-x86_64.AppImage
|
||||||
mv PrismLauncher-*.AppImage.zsync/PrismLauncher-*.AppImage.zsync PrismLauncher-Linux-x86_64.AppImage.zsync
|
|
||||||
mv PrismLauncher-macOS-Legacy*/PrismLauncher.tar.gz PrismLauncher-macOS-Legacy-${{ env.VERSION }}.tar.gz
|
mv PrismLauncher-macOS-Legacy*/PrismLauncher.tar.gz PrismLauncher-macOS-Legacy-${{ env.VERSION }}.tar.gz
|
||||||
mv PrismLauncher-macOS*/PrismLauncher.tar.gz PrismLauncher-macOS-${{ env.VERSION }}.tar.gz
|
mv PrismLauncher-macOS*/PrismLauncher.tar.gz PrismLauncher-macOS-${{ env.VERSION }}.tar.gz
|
||||||
|
|
||||||
@ -80,22 +78,25 @@ jobs:
|
|||||||
- name: Create release
|
- name: Create release
|
||||||
id: create_release
|
id: create_release
|
||||||
uses: softprops/action-gh-release@v1
|
uses: softprops/action-gh-release@v1
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
with:
|
with:
|
||||||
token: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
tag_name: ${{ github.ref }}
|
tag_name: ${{ github.ref }}
|
||||||
name: Prism Launcher ${{ env.VERSION }}
|
name: Prism Launcher ${{ env.VERSION }}
|
||||||
draft: true
|
draft: true
|
||||||
prerelease: false
|
prerelease: false
|
||||||
files: |
|
files: |
|
||||||
PrismLauncher-Linux-Qt5-${{ env.VERSION }}.tar.gz
|
PrismLauncher-Linux-${{ env.VERSION }}.tar.gz
|
||||||
PrismLauncher-Linux-Qt5-Portable-${{ env.VERSION }}.tar.gz
|
PrismLauncher-Linux-Portable-${{ env.VERSION }}.tar.gz
|
||||||
PrismLauncher-Linux-x86_64.AppImage
|
PrismLauncher-Linux-${{ env.VERSION }}-x86_64.AppImage
|
||||||
PrismLauncher-Linux-x86_64.AppImage.zsync
|
|
||||||
PrismLauncher-Linux-Qt6-${{ env.VERSION }}.tar.gz
|
PrismLauncher-Linux-Qt6-${{ env.VERSION }}.tar.gz
|
||||||
PrismLauncher-Linux-Qt6-Portable-${{ env.VERSION }}.tar.gz
|
PrismLauncher-Linux-Qt6-Portable-${{ env.VERSION }}.tar.gz
|
||||||
PrismLauncher-Windows-MinGW-w64-${{ env.VERSION }}.zip
|
PrismLauncher-Windows-MinGW-w64-${{ env.VERSION }}.zip
|
||||||
PrismLauncher-Windows-MinGW-w64-Portable-${{ env.VERSION }}.zip
|
PrismLauncher-Windows-MinGW-w64-Portable-${{ env.VERSION }}.zip
|
||||||
PrismLauncher-Windows-MinGW-w64-Setup-${{ env.VERSION }}.exe
|
PrismLauncher-Windows-MinGW-w64-Setup-${{ env.VERSION }}.exe
|
||||||
|
PrismLauncher-Windows-MSVC-Legacy-${{ env.VERSION }}.zip
|
||||||
|
PrismLauncher-Windows-MSVC-Legacy-Portable-${{ env.VERSION }}.zip
|
||||||
|
PrismLauncher-Windows-MSVC-Legacy-Setup-${{ env.VERSION }}.exe
|
||||||
PrismLauncher-Windows-MSVC-arm64-${{ env.VERSION }}.zip
|
PrismLauncher-Windows-MSVC-arm64-${{ env.VERSION }}.zip
|
||||||
PrismLauncher-Windows-MSVC-arm64-Portable-${{ env.VERSION }}.zip
|
PrismLauncher-Windows-MSVC-arm64-Portable-${{ env.VERSION }}.zip
|
||||||
PrismLauncher-Windows-MSVC-arm64-Setup-${{ env.VERSION }}.exe
|
PrismLauncher-Windows-MSVC-arm64-Setup-${{ env.VERSION }}.exe
|
||||||
|
8
.github/workflows/update-flake.yml
vendored
8
.github/workflows/update-flake.yml
vendored
@ -16,15 +16,13 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v3
|
||||||
- uses: cachix/install-nix-action@6a9a9e84a173d90b3ffb42c5ddaf9ea033fad011 # v23
|
- uses: cachix/install-nix-action@v22
|
||||||
|
|
||||||
- uses: DeterminateSystems/update-flake-lock@v20
|
- uses: DeterminateSystems/update-flake-lock@v19
|
||||||
with:
|
with:
|
||||||
commit-msg: "chore(nix): update lockfile"
|
commit-msg: "chore(nix): update lockfile"
|
||||||
pr-title: "chore(nix): update lockfile"
|
pr-title: "chore(nix): update lockfile"
|
||||||
pr-labels: |
|
pr-labels: |
|
||||||
Linux
|
Linux
|
||||||
packaging
|
|
||||||
simple change
|
simple change
|
||||||
changelog:omit
|
|
||||||
|
@ -33,13 +33,6 @@ if(MSVC)
|
|||||||
# Use /W4 as /Wall includes unnesserey warnings such as added padding to structs
|
# Use /W4 as /Wall includes unnesserey warnings such as added padding to structs
|
||||||
set(CMAKE_CXX_FLAGS "/GS /permissive- /W4 ${CMAKE_CXX_FLAGS}")
|
set(CMAKE_CXX_FLAGS "/GS /permissive- /W4 ${CMAKE_CXX_FLAGS}")
|
||||||
|
|
||||||
# /EHs Enables stack unwind semantics for standard C++ exceptions to ensure stackframes are unwound
|
|
||||||
# and object deconstructors are called when an exception is caught.
|
|
||||||
# without it memory leaks and a warning is printed
|
|
||||||
# /EHc tells the compiler to assume that functions declared as extern "C" never throw a C++ exception
|
|
||||||
# This appears to not always be a defualt compiler option in CMAKE
|
|
||||||
set(CMAKE_CXX_FLAGS "/EHsc ${CMAKE_CXX_FLAGS}")
|
|
||||||
|
|
||||||
# LINK accepts /SUBSYSTEM whics sets if we are a WINDOWS (gui) or a CONSOLE programs
|
# LINK accepts /SUBSYSTEM whics sets if we are a WINDOWS (gui) or a CONSOLE programs
|
||||||
# This implicitly selects an entrypoint specific to the subsystem selected
|
# This implicitly selects an entrypoint specific to the subsystem selected
|
||||||
# qtmain/QtEntryPointLib provides the correct entrypoint (wWinMain) for gui programs
|
# qtmain/QtEntryPointLib provides the correct entrypoint (wWinMain) for gui programs
|
||||||
@ -92,39 +85,38 @@ set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -DTOML_ENABLE_FLOAT16=0")
|
|||||||
# set CXXFLAGS for build targets
|
# set CXXFLAGS for build targets
|
||||||
set(CMAKE_CXX_FLAGS_RELEASE "-O2 -D_FORTIFY_SOURCE=2 ${CMAKE_CXX_FLAGS_RELEASE}")
|
set(CMAKE_CXX_FLAGS_RELEASE "-O2 -D_FORTIFY_SOURCE=2 ${CMAKE_CXX_FLAGS_RELEASE}")
|
||||||
|
|
||||||
option(DEBUG_ADDRESS_SANITIZER "Enable Address Sanitizer in Debug builds" OFF)
|
option(DEBUG_ADDRESS_SANITIZER "Enable Address Sanitizer in Debug builds" on)
|
||||||
|
|
||||||
# If this is a Debug build turn on address sanitiser
|
# If this is a Debug build turn on address sanitiser
|
||||||
if ((CMAKE_BUILD_TYPE STREQUAL "Debug" OR CMAKE_BUILD_TYPE STREQUAL "RelWithDebInfo") AND DEBUG_ADDRESS_SANITIZER)
|
if (CMAKE_BUILD_TYPE STREQUAL "Debug" AND DEBUG_ADDRESS_SANITIZER)
|
||||||
message(STATUS "Address Sanitizer enabled for Debug builds, Turn it off with -DDEBUG_ADDRESS_SANITIZER=off")
|
message(STATUS "Address Sanitizer enabled for Debug builds, Turn it off with -DDEBUG_ADDRESS_SANITIZER=off")
|
||||||
if ("${CMAKE_CXX_COMPILER_ID}" MATCHES "Clang")
|
if ("${CMAKE_CXX_COMPILER_ID}" MATCHES "Clang")
|
||||||
if (CMAKE_CXX_COMPILER_FRONTEND_VARIANT STREQUAL "MSVC")
|
if (CMAKE_CXX_COMPILER_FRONTEND_VARIANT STREQUAL "MSVC")
|
||||||
# using clang with clang-cl front end
|
# using clang with clang-cl front end
|
||||||
message(STATUS "Address Sanitizer available on Clang MSVC frontend")
|
message(STATUS "Address Sanitizer available on Clang MSVC frontend")
|
||||||
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /fsanitize=address /Oy-")
|
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /fsanitize=address /O1 /Oy-")
|
||||||
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} /fsanitize=address /Oy-")
|
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} /fsanitize=address /O1 /Oy-")
|
||||||
else()
|
else()
|
||||||
# AppleClang and Clang
|
# AppleClang and Clang
|
||||||
message(STATUS "Address Sanitizer available on Clang")
|
message(STATUS "Address Sanitizer available on Clang")
|
||||||
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=address -fno-omit-frame-pointer")
|
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=address -O1 -fno-omit-frame-pointer")
|
||||||
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fsanitize=address -fno-omit-frame-pointer")
|
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fsanitize=address -O1 -fno-omit-frame-pointer")
|
||||||
endif()
|
endif()
|
||||||
elseif ("${CMAKE_CXX_COMPILER_ID}" STREQUAL "GNU")
|
elseif ("${CMAKE_CXX_COMPILER_ID}" STREQUAL "GNU")
|
||||||
# GCC
|
# GCC
|
||||||
message(STATUS "Address Sanitizer available on GCC")
|
message(STATUS "Address Sanitizer available on GCC")
|
||||||
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=address -fno-omit-frame-pointer")
|
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=address -O1 -fno-omit-frame-pointer")
|
||||||
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fsanitize=address -fno-omit-frame-pointer")
|
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fsanitize=address -O1 -fno-omit-frame-pointer")
|
||||||
link_libraries("asan")
|
link_libraries("asan")
|
||||||
elseif ("${CMAKE_CXX_COMPILER_ID}" STREQUAL "MSVC")
|
elseif ("${CMAKE_CXX_COMPILER_ID}" STREQUAL "MSVC")
|
||||||
message(STATUS "Address Sanitizer available on MSVC")
|
message(STATUS "Address Sanitizer available on MSVC")
|
||||||
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /fsanitize=address /Oy-")
|
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} /fsanitize=address /O1 /Oy-")
|
||||||
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} /fsanitize=address /Oy-")
|
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} /fsanitize=address /O1 /Oy-")
|
||||||
else()
|
else()
|
||||||
message(STATUS "Address Sanitizer not available on compiler ${CMAKE_CXX_COMPILER_ID}")
|
message(STATUS "Address Sanitizer not available on compiler ${CMAKE_CXX_COMPILER_ID}")
|
||||||
endif()
|
endif()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
|
||||||
option(ENABLE_LTO "Enable Link Time Optimization" off)
|
option(ENABLE_LTO "Enable Link Time Optimization" off)
|
||||||
|
|
||||||
if(ENABLE_LTO)
|
if(ENABLE_LTO)
|
||||||
@ -178,8 +170,8 @@ set(Launcher_NEWS_OPEN_URL "https://prismlauncher.org/news" CACHE STRING "URL th
|
|||||||
set(Launcher_HELP_URL "https://prismlauncher.org/wiki/help-pages/%1" CACHE STRING "URL (with arg %1 to be substituted with page-id) that gets opened when the user requests help")
|
set(Launcher_HELP_URL "https://prismlauncher.org/wiki/help-pages/%1" CACHE STRING "URL (with arg %1 to be substituted with page-id) that gets opened when the user requests help")
|
||||||
|
|
||||||
######## Set version numbers ########
|
######## Set version numbers ########
|
||||||
set(Launcher_VERSION_MAJOR 8)
|
set(Launcher_VERSION_MAJOR 7)
|
||||||
set(Launcher_VERSION_MINOR 0)
|
set(Launcher_VERSION_MINOR 2)
|
||||||
|
|
||||||
set(Launcher_VERSION_NAME "${Launcher_VERSION_MAJOR}.${Launcher_VERSION_MINOR}")
|
set(Launcher_VERSION_NAME "${Launcher_VERSION_MAJOR}.${Launcher_VERSION_MINOR}")
|
||||||
set(Launcher_VERSION_NAME4 "${Launcher_VERSION_MAJOR}.${Launcher_VERSION_MINOR}.0.0")
|
set(Launcher_VERSION_NAME4 "${Launcher_VERSION_MAJOR}.${Launcher_VERSION_MINOR}.0.0")
|
||||||
@ -188,11 +180,8 @@ set(Launcher_VERSION_NAME4_COMMA "${Launcher_VERSION_MAJOR},${Launcher_VERSION_M
|
|||||||
# Build platform.
|
# 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.")
|
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.")
|
||||||
|
|
||||||
# Github repo URL with releases for updater
|
# Channel list URL
|
||||||
set(Launcher_UPDATER_GITHUB_REPO "https://github.com/PrismLauncher/PrismLauncher" CACHE STRING "Base github URL for the updater.")
|
set(Launcher_UPDATER_BASE "" CACHE STRING "Base 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
|
# The metadata server
|
||||||
set(Launcher_META_URL "https://meta.prismlauncher.org/v1/" CACHE STRING "URL to fetch Launcher's meta files from.")
|
set(Launcher_META_URL "https://meta.prismlauncher.org/v1/" CACHE STRING "URL to fetch Launcher's meta files from.")
|
||||||
@ -219,18 +208,6 @@ set(Launcher_SUBREDDIT_URL "https://prismlauncher.org/reddit" CACHE STRING "URL
|
|||||||
set(Launcher_FORCE_BUNDLED_LIBS OFF CACHE BOOL "Prevent using system libraries, if they are available as submodules")
|
set(Launcher_FORCE_BUNDLED_LIBS OFF CACHE BOOL "Prevent using system libraries, if they are available as submodules")
|
||||||
set(Launcher_QT_VERSION_MAJOR "6" CACHE STRING "Major Qt version to build against")
|
set(Launcher_QT_VERSION_MAJOR "6" CACHE STRING "Major Qt version to build against")
|
||||||
|
|
||||||
# Native libraries
|
|
||||||
if(UNIX AND APPLE)
|
|
||||||
set(Launcher_GLFW_LIBRARY_NAME "libglfw.dylib" CACHE STRING "Name of native glfw library")
|
|
||||||
set(Launcher_OPENAL_LIBRARY_NAME "libopenal.dylib" CACHE STRING "Name of native openal library")
|
|
||||||
elseif(UNIX)
|
|
||||||
set(Launcher_GLFW_LIBRARY_NAME "libglfw.so" CACHE STRING "Name of native glfw library")
|
|
||||||
set(Launcher_OPENAL_LIBRARY_NAME "libopenal.so" CACHE STRING "Name of native openal library")
|
|
||||||
elseif(WIN32)
|
|
||||||
set(Launcher_GLFW_LIBRARY_NAME "glfw.dll" CACHE STRING "Name of native glfw library")
|
|
||||||
set(Launcher_OPENAL_LIBRARY_NAME "OpenAL.dll" CACHE STRING "Name of native openal library")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# API Keys
|
# API Keys
|
||||||
# NOTE: These API keys are here for convenience. If you rebrand this software or intend to break the terms of service
|
# NOTE: These API keys are here for convenience. If you rebrand this software or intend to break the terms of service
|
||||||
# of these platforms, please change these API keys beforehand.
|
# of these platforms, please change these API keys beforehand.
|
||||||
@ -248,11 +225,6 @@ set(Launcher_MSA_CLIENT_ID "c36a9fb6-4f2a-41ff-90bd-ae7cc92031eb" CACHE STRING "
|
|||||||
# This key was issued specifically for Prism Launcher
|
# This key was issued specifically for Prism Launcher
|
||||||
set(Launcher_CURSEFORGE_API_KEY "$2a$10$wuAJuNZuted3NORVmpgUC.m8sI.pv1tOPKZyBgLFGjxFp/br0lZCC" CACHE STRING "API key for the CurseForge platform")
|
set(Launcher_CURSEFORGE_API_KEY "$2a$10$wuAJuNZuted3NORVmpgUC.m8sI.pv1tOPKZyBgLFGjxFp/br0lZCC" CACHE STRING "API key for the CurseForge platform")
|
||||||
|
|
||||||
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
|
#### Check the current Git commit and branch
|
||||||
include(GetGitRevisionDescription)
|
include(GetGitRevisionDescription)
|
||||||
@ -346,13 +318,6 @@ add_subdirectory(program_info)
|
|||||||
|
|
||||||
####################################### Install layout #######################################
|
####################################### 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))
|
if(NOT (UNIX AND APPLE))
|
||||||
# Install "portable.txt" if selected component is "portable"
|
# Install "portable.txt" if selected component is "portable"
|
||||||
install(FILES "${CMAKE_CURRENT_SOURCE_DIR}/${Launcher_Portable_File}" DESTINATION "." COMPONENT portable EXCLUDE_FROM_ALL)
|
install(FILES "${CMAKE_CURRENT_SOURCE_DIR}/${Launcher_Portable_File}" DESTINATION "." COMPONENT portable EXCLUDE_FROM_ALL)
|
||||||
@ -377,9 +342,9 @@ if(UNIX AND APPLE)
|
|||||||
set(MACOSX_BUNDLE_SHORT_VERSION_STRING "${Launcher_VERSION_NAME}")
|
set(MACOSX_BUNDLE_SHORT_VERSION_STRING "${Launcher_VERSION_NAME}")
|
||||||
set(MACOSX_BUNDLE_LONG_VERSION_STRING "${Launcher_VERSION_NAME}")
|
set(MACOSX_BUNDLE_LONG_VERSION_STRING "${Launcher_VERSION_NAME}")
|
||||||
set(MACOSX_BUNDLE_ICON_FILE ${Launcher_Name}.icns)
|
set(MACOSX_BUNDLE_ICON_FILE ${Launcher_Name}.icns)
|
||||||
set(MACOSX_BUNDLE_COPYRIGHT "© 2022-2023 ${Launcher_Copyright_Mac}")
|
set(MACOSX_BUNDLE_COPYRIGHT "© 2022 ${Launcher_Copyright_Mac}")
|
||||||
set(MACOSX_SPARKLE_UPDATE_PUBLIC_KEY "v55ZWWD6QlPoXGV6VLzOTZxZUggWeE51X8cRQyQh6vA=" CACHE STRING "Public key for Sparkle update feed")
|
set(MACOSX_SPARKLE_UPDATE_PUBLIC_KEY "v55ZWWD6QlPoXGV6VLzOTZxZUggWeE51X8cRQyQh6vA=")
|
||||||
set(MACOSX_SPARKLE_UPDATE_FEED_URL "https://prismlauncher.org/feed/appcast.xml" CACHE STRING "URL for Sparkle update feed")
|
set(MACOSX_SPARKLE_UPDATE_FEED_URL "https://prismlauncher.org/feed/appcast.xml")
|
||||||
|
|
||||||
set(MACOSX_SPARKLE_DOWNLOAD_URL "https://github.com/sparkle-project/Sparkle/releases/download/2.1.0/Sparkle-2.1.0.tar.xz" CACHE STRING "URL to Sparkle release archive")
|
set(MACOSX_SPARKLE_DOWNLOAD_URL "https://github.com/sparkle-project/Sparkle/releases/download/2.1.0/Sparkle-2.1.0.tar.xz" CACHE STRING "URL to Sparkle release archive")
|
||||||
set(MACOSX_SPARKLE_SHA256 "bf6ac1caa9f8d321d5784859c88da874f28412f37fb327bc21b7b14c5d61ef94" CACHE STRING "SHA256 checksum for Sparkle release archive")
|
set(MACOSX_SPARKLE_SHA256 "bf6ac1caa9f8d321d5784859c88da874f28412f37fb327bc21b7b14c5d61ef94" CACHE STRING "SHA256 checksum for Sparkle release archive")
|
||||||
@ -388,12 +353,8 @@ if(UNIX AND APPLE)
|
|||||||
# directories to look for dependencies
|
# directories to look for dependencies
|
||||||
set(DIRS ${QT_LIBS_DIR} ${QT_LIBEXECS_DIR} ${CMAKE_LIBRARY_OUTPUT_DIRECTORY} ${CMAKE_RUNTIME_OUTPUT_DIRECTORY} ${MACOSX_SPARKLE_DIR})
|
set(DIRS ${QT_LIBS_DIR} ${QT_LIBEXECS_DIR} ${CMAKE_LIBRARY_OUTPUT_DIRECTORY} ${CMAKE_RUNTIME_OUTPUT_DIRECTORY} ${MACOSX_SPARKLE_DIR})
|
||||||
|
|
||||||
if(NOT MACOSX_SPARKLE_UPDATE_PUBLIC_KEY STREQUAL "" AND NOT MACOSX_SPARKLE_UPDATE_FEED_URL STREQUAL "")
|
|
||||||
set(Launcher_ENABLE_UPDATER YES)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# install as bundle
|
# install as bundle
|
||||||
set(INSTALL_BUNDLE "full" CACHE STRING "Use fixup_bundle to bundle dependencies")
|
set(INSTALL_BUNDLE "full")
|
||||||
|
|
||||||
# Add the icon
|
# Add the icon
|
||||||
install(FILES ${Launcher_Branding_ICNS} DESTINATION ${RESOURCES_DEST_DIR} RENAME ${Launcher_Name}.icns)
|
install(FILES ${Launcher_Branding_ICNS} DESTINATION ${RESOURCES_DEST_DIR} RENAME ${Launcher_Name}.icns)
|
||||||
@ -406,7 +367,7 @@ elseif(UNIX)
|
|||||||
set(JARS_DEST_DIR "share/${Launcher_Name}")
|
set(JARS_DEST_DIR "share/${Launcher_Name}")
|
||||||
|
|
||||||
# install as bundle with no dependencies included
|
# install as bundle with no dependencies included
|
||||||
set(INSTALL_BUNDLE "nodeps" CACHE STRING "Use fixup_bundle to bundle dependencies")
|
set(INSTALL_BUNDLE "nodeps")
|
||||||
|
|
||||||
# Set RPATH
|
# Set RPATH
|
||||||
SET(Launcher_BINARY_RPATH "$ORIGIN/")
|
SET(Launcher_BINARY_RPATH "$ORIGIN/")
|
||||||
@ -440,7 +401,7 @@ elseif(WIN32)
|
|||||||
set(DIRS ${QT_LIBS_DIR} ${QT_LIBEXECS_DIR} ${CMAKE_LIBRARY_OUTPUT_DIRECTORY} ${CMAKE_RUNTIME_OUTPUT_DIRECTORY})
|
set(DIRS ${QT_LIBS_DIR} ${QT_LIBEXECS_DIR} ${CMAKE_LIBRARY_OUTPUT_DIRECTORY} ${CMAKE_RUNTIME_OUTPUT_DIRECTORY})
|
||||||
|
|
||||||
# install as bundle
|
# install as bundle
|
||||||
set(INSTALL_BUNDLE "full" CACHE STRING "Use fixup_bundle to bundle dependencies")
|
set(INSTALL_BUNDLE "full")
|
||||||
else()
|
else()
|
||||||
message(FATAL_ERROR "Platform not supported")
|
message(FATAL_ERROR "Platform not supported")
|
||||||
endif()
|
endif()
|
||||||
|
@ -19,7 +19,7 @@ In an effort to ensure that the code you contribute is actually compatible with
|
|||||||
|
|
||||||
This can be done by appending `-s` to your `git commit` call, or by manually appending the following text to your commit message:
|
This can be done by appending `-s` to your `git commit` call, or by manually appending the following text to your commit message:
|
||||||
|
|
||||||
```text
|
```
|
||||||
<commit message>
|
<commit message>
|
||||||
|
|
||||||
Signed-off-by: Author name <Author email>
|
Signed-off-by: Author name <Author email>
|
||||||
@ -27,7 +27,7 @@ Signed-off-by: Author name <Author email>
|
|||||||
|
|
||||||
By signing off your work, you agree to the terms below:
|
By signing off your work, you agree to the terms below:
|
||||||
|
|
||||||
```text
|
```
|
||||||
Developer's Certificate of Origin 1.1
|
Developer's Certificate of Origin 1.1
|
||||||
|
|
||||||
By making a contribution to this project, I certify that:
|
By making a contribution to this project, I certify that:
|
||||||
@ -61,9 +61,3 @@ As a bonus, you can also [cryptographically sign your commits][gh-signing-commit
|
|||||||
|
|
||||||
[gh-signing-commits]: https://docs.github.com/en/authentication/managing-commit-signature-verification/signing-commits
|
[gh-signing-commits]: https://docs.github.com/en/authentication/managing-commit-signature-verification/signing-commits
|
||||||
[gh-vigilant-mode]: https://docs.github.com/en/authentication/managing-commit-signature-verification/displaying-verification-statuses-for-all-of-your-commits
|
[gh-vigilant-mode]: https://docs.github.com/en/authentication/managing-commit-signature-verification/displaying-verification-statuses-for-all-of-your-commits
|
||||||
|
|
||||||
## Backporting to Release Branches
|
|
||||||
|
|
||||||
We use [automated backports](https://github.com/PrismLauncher/PrismLauncher/blob/develop/.github/workflows/backport.yml) to merge specific contributions from develop into `release` branches.
|
|
||||||
|
|
||||||
This is done when pull requests are merged and have labels such as `backport release-7.x` - which should be added along with the milestone for the release.
|
|
||||||
|
15
README.md
15
README.md
@ -18,18 +18,11 @@
|
|||||||
</a>
|
</a>
|
||||||
|
|
||||||
- All downloads and instructions for Prism Launcher can be found on our [Website](https://prismlauncher.org/download).
|
- All downloads and instructions for Prism Launcher can be found on our [Website](https://prismlauncher.org/download).
|
||||||
- Last build status can be found in the [GitHub Actions](https://github.com/PrismLauncher/PrismLauncher/actions) tab (this also includes the pull requests status).
|
- Last build status can be found in the [GitHub Actions](https://github.com/PrismLauncher/PrismLauncher/actions).
|
||||||
|
|
||||||
### Development Builds
|
### Development Builds
|
||||||
|
|
||||||
Please understand that these builds are not intended for most users. There may be bugs, and other instabilities. You have been warned.
|
There are development builds available [here](https://github.com/PrismLauncher/PrismLauncher/actions). These have debug information in the binaries, so their file sizes are relatively larger.
|
||||||
|
|
||||||
There are development builds available through:
|
|
||||||
|
|
||||||
- [GitHub Actions](https://github.com/PrismLauncher/PrismLauncher/actions) (includes builds from pull requests opened by contribuitors)
|
|
||||||
- [nightly.link](https://nightly.link/PrismLauncher/PrismLauncher/workflows/trigger_builds/develop) (this will always point only to the latest version of develop)
|
|
||||||
|
|
||||||
These have debug information in the binaries, so their file sizes are relatively larger.
|
|
||||||
|
|
||||||
Prebuilt Development builds are provided for **Linux**, **Windows** and **macOS**.
|
Prebuilt Development builds are provided for **Linux**, **Windows** and **macOS**.
|
||||||
|
|
||||||
@ -37,7 +30,7 @@ For **Arch**, **Debian**, **Fedora**, **OpenSUSE (Tumbleweed)** and **Gentoo**,
|
|||||||
|
|
||||||
[![prismlauncher-git](https://img.shields.io/badge/aur-prismlauncher--git-1793D1?label=AUR&logo=archlinux&logoColor=white)](https://aur.archlinux.org/packages/prismlauncher-git) [![prismlauncher-git](https://img.shields.io/badge/aur-prismlauncher--qt5--git-1793D1?label=AUR&logo=archlinux&logoColor=white)](https://aur.archlinux.org/packages/prismlauncher-qt5-git) [![prismlauncher-git](https://img.shields.io/badge/mpr-prismlauncher--git-A80030?label=MPR&logo=debian&logoColor=white)](https://mpr.makedeb.org/packages/prismlauncher-git)<br />[![prismlauncher-nightly](https://img.shields.io/badge/copr-prismlauncher--nightly-51A2DA?label=COPR&logo=fedora&logoColor=white)](https://copr.fedorainfracloud.org/coprs/g3tchoo/prismlauncher/) [![prismlauncher-nightly](https://img.shields.io/badge/OBS-prismlauncher--nightly-3AB6A9?logo=opensuse&logoColor=white)](https://build.opensuse.org/project/show/home:getchoo) [![prismlauncher-9999](https://img.shields.io/badge/gentoo-prismlauncher--9999-4D4270?label=Gentoo&logo=gentoo&logoColor=white)](https://packages.gentoo.org/packages/games-action/prismlauncher)
|
[![prismlauncher-git](https://img.shields.io/badge/aur-prismlauncher--git-1793D1?label=AUR&logo=archlinux&logoColor=white)](https://aur.archlinux.org/packages/prismlauncher-git) [![prismlauncher-git](https://img.shields.io/badge/aur-prismlauncher--qt5--git-1793D1?label=AUR&logo=archlinux&logoColor=white)](https://aur.archlinux.org/packages/prismlauncher-qt5-git) [![prismlauncher-git](https://img.shields.io/badge/mpr-prismlauncher--git-A80030?label=MPR&logo=debian&logoColor=white)](https://mpr.makedeb.org/packages/prismlauncher-git)<br />[![prismlauncher-nightly](https://img.shields.io/badge/copr-prismlauncher--nightly-51A2DA?label=COPR&logo=fedora&logoColor=white)](https://copr.fedorainfracloud.org/coprs/g3tchoo/prismlauncher/) [![prismlauncher-nightly](https://img.shields.io/badge/OBS-prismlauncher--nightly-3AB6A9?logo=opensuse&logoColor=white)](https://build.opensuse.org/project/show/home:getchoo) [![prismlauncher-9999](https://img.shields.io/badge/gentoo-prismlauncher--9999-4D4270?label=Gentoo&logo=gentoo&logoColor=white)](https://packages.gentoo.org/packages/games-action/prismlauncher)
|
||||||
|
|
||||||
These packages are also available to all the distributions based on the ones mentioned above.
|
These packages are also availiable to all the distributions based on the ones mentioned above.
|
||||||
|
|
||||||
## Community & Support
|
## Community & Support
|
||||||
|
|
||||||
@ -57,7 +50,7 @@ Feel free to create a GitHub issue if you find a bug or want to suggest a new fe
|
|||||||
|
|
||||||
## Translations
|
## Translations
|
||||||
|
|
||||||
The translation effort for Prism Launcher is hosted on [Weblate](https://hosted.weblate.org/projects/prismlauncher/launcher/) and information about translating Prism Launcher is available at <https://github.com/PrismLauncher/Translations>.
|
The translation effort for Prism Launcher is hosted on [Weblate](https://hosted.weblate.org/projects/prismlauncher/launcher/) and information about translating Prism Launcher is available at <https://github.com/PrismLauncher/Translations>
|
||||||
|
|
||||||
## Building
|
## Building
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
@ -33,7 +33,6 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <qstringliteral.h>
|
|
||||||
#include "BuildConfig.h"
|
#include "BuildConfig.h"
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
|
|
||||||
@ -60,16 +59,8 @@ Config::Config()
|
|||||||
VERSION_MINOR = @Launcher_VERSION_MINOR@;
|
VERSION_MINOR = @Launcher_VERSION_MINOR@;
|
||||||
|
|
||||||
BUILD_PLATFORM = "@Launcher_BUILD_PLATFORM@";
|
BUILD_PLATFORM = "@Launcher_BUILD_PLATFORM@";
|
||||||
BUILD_ARTIFACT = "@Launcher_BUILD_ARTIFACT@";
|
|
||||||
BUILD_DATE = "@Launcher_BUILD_TIMESTAMP@";
|
BUILD_DATE = "@Launcher_BUILD_TIMESTAMP@";
|
||||||
UPDATER_GITHUB_REPO = "@Launcher_UPDATER_GITHUB_REPO@";
|
UPDATER_BASE = "@Launcher_UPDATER_BASE@";
|
||||||
|
|
||||||
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_PUB_KEY = "@MACOSX_SPARKLE_UPDATE_PUBLIC_KEY@";
|
||||||
MAC_SPARKLE_APPCAST_URL = "@MACOSX_SPARKLE_UPDATE_FEED_URL@";
|
MAC_SPARKLE_APPCAST_URL = "@MACOSX_SPARKLE_UPDATE_FEED_URL@";
|
||||||
@ -77,8 +68,6 @@ Config::Config()
|
|||||||
if (!MAC_SPARKLE_PUB_KEY.isEmpty() && !MAC_SPARKLE_APPCAST_URL.isEmpty())
|
if (!MAC_SPARKLE_PUB_KEY.isEmpty() && !MAC_SPARKLE_APPCAST_URL.isEmpty())
|
||||||
{
|
{
|
||||||
UPDATER_ENABLED = true;
|
UPDATER_ENABLED = true;
|
||||||
} else if(!UPDATER_GITHUB_REPO.isEmpty() && !BUILD_ARTIFACT.isEmpty()) {
|
|
||||||
UPDATER_ENABLED = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
GIT_COMMIT = "@Launcher_GIT_COMMIT@";
|
GIT_COMMIT = "@Launcher_GIT_COMMIT@";
|
||||||
@ -99,7 +88,10 @@ Config::Config()
|
|||||||
if (GIT_REFSPEC.startsWith("refs/heads/"))
|
if (GIT_REFSPEC.startsWith("refs/heads/"))
|
||||||
{
|
{
|
||||||
VERSION_CHANNEL = GIT_REFSPEC;
|
VERSION_CHANNEL = GIT_REFSPEC;
|
||||||
VERSION_CHANNEL.remove("refs/heads/");
|
VERSION_CHANNEL.remove("refs/heads/");
|
||||||
|
if(!UPDATER_BASE.isEmpty() && !BUILD_PLATFORM.isEmpty()) {
|
||||||
|
UPDATER_ENABLED = true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (!GIT_COMMIT.isEmpty())
|
else if (!GIT_COMMIT.isEmpty())
|
||||||
{
|
{
|
||||||
@ -118,9 +110,6 @@ Config::Config()
|
|||||||
FLAME_API_KEY = "@Launcher_CURSEFORGE_API_KEY@";
|
FLAME_API_KEY = "@Launcher_CURSEFORGE_API_KEY@";
|
||||||
META_URL = "@Launcher_META_URL@";
|
META_URL = "@Launcher_META_URL@";
|
||||||
|
|
||||||
GLFW_LIBRARY_NAME = "@Launcher_GLFW_LIBRARY_NAME@";
|
|
||||||
OPENAL_LIBRARY_NAME = "@Launcher_OPENAL_LIBRARY_NAME@";
|
|
||||||
|
|
||||||
BUG_TRACKER_URL = "@Launcher_BUG_TRACKER_URL@";
|
BUG_TRACKER_URL = "@Launcher_BUG_TRACKER_URL@";
|
||||||
TRANSLATIONS_URL = "@Launcher_TRANSLATIONS_URL@";
|
TRANSLATIONS_URL = "@Launcher_TRANSLATIONS_URL@";
|
||||||
MATRIX_URL = "@Launcher_MATRIX_URL@";
|
MATRIX_URL = "@Launcher_MATRIX_URL@";
|
||||||
@ -144,16 +133,3 @@ QString Config::printableVersionString() const
|
|||||||
}
|
}
|
||||||
return vstr;
|
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);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
@ -36,8 +36,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
#include <QList>
|
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
#include <QList>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* \brief The Config class holds all the build-time information passed from the build system.
|
* \brief The Config class holds all the build-time information passed from the build system.
|
||||||
@ -71,29 +71,11 @@ class Config {
|
|||||||
/// A short string identifying this build's platform or distribution.
|
/// A short string identifying this build's platform or distribution.
|
||||||
QString BUILD_PLATFORM;
|
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
|
/// A string containing the build timestamp
|
||||||
QString BUILD_DATE;
|
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
|
/// URL for the updater's channel
|
||||||
QString UPDATER_GITHUB_REPO;
|
QString UPDATER_BASE;
|
||||||
|
|
||||||
/// The public key used to sign releases for the Sparkle updater appcast
|
/// The public key used to sign releases for the Sparkle updater appcast
|
||||||
QString MAC_SPARKLE_PUB_KEY;
|
QString MAC_SPARKLE_PUB_KEY;
|
||||||
@ -152,9 +134,6 @@ class Config {
|
|||||||
*/
|
*/
|
||||||
QString META_URL;
|
QString META_URL;
|
||||||
|
|
||||||
QString GLFW_LIBRARY_NAME;
|
|
||||||
QString OPENAL_LIBRARY_NAME;
|
|
||||||
|
|
||||||
QString BUG_TRACKER_URL;
|
QString BUG_TRACKER_URL;
|
||||||
QString TRANSLATIONS_URL;
|
QString TRANSLATIONS_URL;
|
||||||
QString MATRIX_URL;
|
QString MATRIX_URL;
|
||||||
@ -166,7 +145,7 @@ class Config {
|
|||||||
QString AUTH_BASE = "https://authserver.mojang.com/";
|
QString AUTH_BASE = "https://authserver.mojang.com/";
|
||||||
QString IMGUR_BASE_URL = "https://api.imgur.com/3/";
|
QString IMGUR_BASE_URL = "https://api.imgur.com/3/";
|
||||||
QString FMLLIBS_BASE_URL = "https://files.prismlauncher.org/fmllibs/"; // FIXME: move into CMakeLists
|
QString FMLLIBS_BASE_URL = "https://files.prismlauncher.org/fmllibs/"; // FIXME: move into CMakeLists
|
||||||
QString TRANSLATIONS_BASE_URL = "https://i18n.prismlauncher.org/"; // FIXME: move into CMakeLists
|
QString TRANSLATIONS_BASE_URL = "https://i18n.prismlauncher.org/"; // FIXME: move into CMakeLists
|
||||||
|
|
||||||
QString MODPACKSCH_API_BASE_URL = "https://api.modpacks.ch/";
|
QString MODPACKSCH_API_BASE_URL = "https://api.modpacks.ch/";
|
||||||
|
|
||||||
@ -183,7 +162,7 @@ class Config {
|
|||||||
|
|
||||||
QString MODRINTH_STAGING_URL = "https://staging-api.modrinth.com/v2";
|
QString MODRINTH_STAGING_URL = "https://staging-api.modrinth.com/v2";
|
||||||
QString MODRINTH_PROD_URL = "https://api.modrinth.com/v2";
|
QString MODRINTH_PROD_URL = "https://api.modrinth.com/v2";
|
||||||
QStringList MODRINTH_MRPACK_HOSTS{ "cdn.modrinth.com", "github.com", "raw.githubusercontent.com", "gitlab.com" };
|
QStringList MODRINTH_MRPACK_HOSTS{"cdn.modrinth.com", "github.com", "raw.githubusercontent.com", "gitlab.com"};
|
||||||
|
|
||||||
QString FLAME_BASE_URL = "https://api.curseforge.com/v1";
|
QString FLAME_BASE_URL = "https://api.curseforge.com/v1";
|
||||||
|
|
||||||
@ -193,18 +172,6 @@ class Config {
|
|||||||
* \return The version number in string format (major.minor.revision.build).
|
* \return The version number in string format (major.minor.revision.build).
|
||||||
*/
|
*/
|
||||||
QString printableVersionString() const;
|
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;
|
extern const Config BuildConfig;
|
||||||
|
@ -1,155 +0,0 @@
|
|||||||
#
|
|
||||||
# Function to set compiler warnings with reasonable defaults at the project level.
|
|
||||||
# Taken from https://github.com/aminya/project_options/blob/main/src/CompilerWarnings.cmake
|
|
||||||
# under the folowing license:
|
|
||||||
#
|
|
||||||
# MIT License
|
|
||||||
#
|
|
||||||
# Copyright (c) 2022-2100 Amin Yahyaabadi
|
|
||||||
#
|
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
# of this software and associated documentation files (the "Software"), to deal
|
|
||||||
# in the Software without restriction, including without limitation the rights
|
|
||||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
# copies of the Software, and to permit persons to whom the Software is
|
|
||||||
# furnished to do so, subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be included in all
|
|
||||||
# copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
# SOFTWARE.
|
|
||||||
#
|
|
||||||
|
|
||||||
include_guard()
|
|
||||||
|
|
||||||
function(_set_project_warnings_add_target_link_option TARGET OPTIONS)
|
|
||||||
target_link_options(${_project_name} INTERFACE ${OPTIONS})
|
|
||||||
endfunction()
|
|
||||||
|
|
||||||
# Set the compiler warnings
|
|
||||||
#
|
|
||||||
# https://clang.llvm.org/docs/DiagnosticsReference.html
|
|
||||||
# https://github.com/lefticus/cppbestpractices/blob/master/02-Use_the_Tools_Available.md
|
|
||||||
function(
|
|
||||||
set_project_warnings
|
|
||||||
_project_name
|
|
||||||
MSVC_WARNINGS
|
|
||||||
CLANG_WARNINGS
|
|
||||||
GCC_WARNINGS
|
|
||||||
)
|
|
||||||
if("${MSVC_WARNINGS}" STREQUAL "")
|
|
||||||
set(MSVC_WARNINGS
|
|
||||||
/W4 # Baseline reasonable warnings
|
|
||||||
/w14242 # 'identifier': conversion from 'type1' to 'type1', possible loss of data
|
|
||||||
/w14254 # 'operator': conversion from 'type1:field_bits' to 'type2:field_bits', possible loss of data
|
|
||||||
/w14263 # 'function': member function does not override any base class virtual member function
|
|
||||||
/w14265 # 'classname': class has virtual functions, but destructor is not virtual instances of this class may not
|
|
||||||
# be destructed correctly
|
|
||||||
/w14287 # 'operator': unsigned/negative constant mismatch
|
|
||||||
/we4289 # nonstandard extension used: 'variable': loop control variable declared in the for-loop is used outside
|
|
||||||
# the for-loop scope
|
|
||||||
/w14296 # 'operator': expression is always 'boolean_value'
|
|
||||||
/w14311 # 'variable': pointer truncation from 'type1' to 'type2'
|
|
||||||
/w14545 # expression before comma evaluates to a function which is missing an argument list
|
|
||||||
/w14546 # function call before comma missing argument list
|
|
||||||
/w14547 # 'operator': operator before comma has no effect; expected operator with side-effect
|
|
||||||
/w14549 # 'operator': operator before comma has no effect; did you intend 'operator'?
|
|
||||||
/w14555 # expression has no effect; expected expression with side- effect
|
|
||||||
/w14619 # pragma warning: there is no warning number 'number'
|
|
||||||
/w14640 # Enable warning on thread un-safe static member initialization
|
|
||||||
/w14826 # Conversion from 'type1' to 'type_2' is sign-extended. This may cause unexpected runtime behavior.
|
|
||||||
/w14905 # wide string literal cast to 'LPSTR'
|
|
||||||
/w14906 # string literal cast to 'LPWSTR'
|
|
||||||
/w14928 # illegal copy-initialization; more than one user-defined conversion has been implicitly applied
|
|
||||||
/permissive- # standards conformance mode for MSVC compiler.
|
|
||||||
)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if("${CLANG_WARNINGS}" STREQUAL "")
|
|
||||||
set(CLANG_WARNINGS
|
|
||||||
-Wall
|
|
||||||
-Wextra # reasonable and standard
|
|
||||||
-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
|
|
||||||
-Wold-style-cast # warn for c-style casts
|
|
||||||
-Wcast-align # warn for potential performance problem casts
|
|
||||||
-Wunused # warn on anything being unused
|
|
||||||
-Woverloaded-virtual # warn if you overload (not override) a virtual function
|
|
||||||
-Wpedantic # warn if non-standard C++ is used
|
|
||||||
-Wconversion # warn on type conversions that may lose data
|
|
||||||
-Wsign-conversion # warn on sign conversions
|
|
||||||
-Wnull-dereference # warn if a null dereference is detected
|
|
||||||
-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()
|
|
||||||
|
|
||||||
if("${GCC_WARNINGS}" STREQUAL "")
|
|
||||||
set(GCC_WARNINGS
|
|
||||||
${CLANG_WARNINGS}
|
|
||||||
-Wmisleading-indentation # warn if indentation implies blocks where blocks do not exist
|
|
||||||
-Wduplicated-cond # warn if if / else chain has duplicated conditions
|
|
||||||
-Wduplicated-branches # warn if if / else branches have duplicated code
|
|
||||||
-Wlogical-op # warn about logical operations being used where bitwise were probably wanted
|
|
||||||
-Wuseless-cast # warn if you perform a cast to the same type
|
|
||||||
)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(MSVC)
|
|
||||||
set(PROJECT_WARNINGS_CXX ${MSVC_WARNINGS})
|
|
||||||
elseif(CMAKE_CXX_COMPILER_ID MATCHES ".*Clang")
|
|
||||||
set(PROJECT_WARNINGS_CXX ${CLANG_WARNINGS})
|
|
||||||
elseif(CMAKE_CXX_COMPILER_ID STREQUAL "GNU")
|
|
||||||
set(PROJECT_WARNINGS_CXX ${GCC_WARNINGS})
|
|
||||||
else()
|
|
||||||
message(AUTHOR_WARNING "No compiler warnings set for CXX compiler: '${CMAKE_CXX_COMPILER_ID}'")
|
|
||||||
# TODO support Intel compiler
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Add C warnings
|
|
||||||
set(PROJECT_WARNINGS_C "${PROJECT_WARNINGS_CXX}")
|
|
||||||
list(
|
|
||||||
REMOVE_ITEM
|
|
||||||
PROJECT_WARNINGS_C
|
|
||||||
-Wnon-virtual-dtor
|
|
||||||
-Wold-style-cast
|
|
||||||
-Woverloaded-virtual
|
|
||||||
-Wuseless-cast
|
|
||||||
-Wextra-semi
|
|
||||||
)
|
|
||||||
|
|
||||||
target_compile_options(
|
|
||||||
${_project_name}
|
|
||||||
INTERFACE # C++ warnings
|
|
||||||
$<$<COMPILE_LANGUAGE:CXX>:${PROJECT_WARNINGS_CXX}>
|
|
||||||
# C warnings
|
|
||||||
$<$<COMPILE_LANGUAGE:C>:${PROJECT_WARNINGS_C}>
|
|
||||||
)
|
|
||||||
|
|
||||||
# If we are using the compiler as a linker driver pass the warnings to it
|
|
||||||
# (most useful when using LTO or warnings as errors)
|
|
||||||
if(CMAKE_CXX_LINK_EXECUTABLE MATCHES "^<CMAKE_CXX_COMPILER>")
|
|
||||||
_set_project_warnings_add_target_link_option(
|
|
||||||
${_project_name} "$<$<COMPILE_LANGUAGE:CXX>:${PROJECT_WARNINGS_CXX}>"
|
|
||||||
)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(CMAKE_C_LINK_EXECUTABLE MATCHES "^<CMAKE_C_COMPILER>")
|
|
||||||
_set_project_warnings_add_target_link_option(
|
|
||||||
${_project_name} "$<$<COMPILE_LANGUAGE:C>:${PROJECT_WARNINGS_C}>"
|
|
||||||
)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
endfunction()
|
|
@ -67,16 +67,5 @@
|
|||||||
<string>Alternate</string>
|
<string>Alternate</string>
|
||||||
</dict>
|
</dict>
|
||||||
</array>
|
</array>
|
||||||
<key>CFBundleURLTypes</key>
|
|
||||||
<array>
|
|
||||||
<dict>
|
|
||||||
<key>CFBundleURLName</key>
|
|
||||||
<string>Curseforge</string>
|
|
||||||
<key>CFBundleURLSchemes</key>
|
|
||||||
<array>
|
|
||||||
<string>curseforge</string>
|
|
||||||
</array>
|
|
||||||
</dict>
|
|
||||||
</array>
|
|
||||||
</dict>
|
</dict>
|
||||||
</plist>
|
</plist>
|
||||||
|
46
flake.lock
46
flake.lock
@ -3,11 +3,11 @@
|
|||||||
"flake-compat": {
|
"flake-compat": {
|
||||||
"flake": false,
|
"flake": false,
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1696426674,
|
"lastModified": 1673956053,
|
||||||
"narHash": "sha256-kvjfFW7WAETZlt09AgDn1MrtKzP7t90Vf7vypd3OL1U=",
|
"narHash": "sha256-4gtG9iQuiKITOjNQQeQIpoIB6b16fm+504Ch3sNKLd8=",
|
||||||
"owner": "edolstra",
|
"owner": "edolstra",
|
||||||
"repo": "flake-compat",
|
"repo": "flake-compat",
|
||||||
"rev": "0f9255e01c2351cc7d116c072cb317785dd33b33",
|
"rev": "35bb57c0c8d8b62bbfd284272c928ceb64ddbde9",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@ -21,11 +21,11 @@
|
|||||||
"nixpkgs-lib": "nixpkgs-lib"
|
"nixpkgs-lib": "nixpkgs-lib"
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1698882062,
|
"lastModified": 1688466019,
|
||||||
"narHash": "sha256-HkhafUayIqxXyHH1X8d9RDl1M2CkFgZLjKD3MzabiEo=",
|
"narHash": "sha256-VeM2akYrBYMsb4W/MmBo1zmaMfgbL4cH3Pu8PGyIwJ0=",
|
||||||
"owner": "hercules-ci",
|
"owner": "hercules-ci",
|
||||||
"repo": "flake-parts",
|
"repo": "flake-parts",
|
||||||
"rev": "8c9fa2545007b49a5db5f650ae91f227672c3877",
|
"rev": "8e8d955c22df93dbe24f19ea04f47a74adbdc5ec",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@ -89,28 +89,13 @@
|
|||||||
"type": "github"
|
"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": {
|
"nixpkgs": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1699094435,
|
"lastModified": 1690026219,
|
||||||
"narHash": "sha256-YLZ5/KKZ1PyLrm2MO8UxRe4H3M0/oaYqNhSlq6FDeeA=",
|
"narHash": "sha256-oOduRk/kzQxOBknZXTLSEYd7tk+GoKvr8wV6Ab+t4AU=",
|
||||||
"owner": "nixos",
|
"owner": "nixos",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "9d5d25bbfe8c0297ebe85324addcb5020ed1a454",
|
"rev": "f465da166263bc0d4b39dfd4ca28b777c92d4b73",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@ -123,11 +108,11 @@
|
|||||||
"nixpkgs-lib": {
|
"nixpkgs-lib": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"dir": "lib",
|
"dir": "lib",
|
||||||
"lastModified": 1698611440,
|
"lastModified": 1688049487,
|
||||||
"narHash": "sha256-jPjHjrerhYDy3q9+s5EAsuhyhuknNfowY6yt6pjn9pc=",
|
"narHash": "sha256-100g4iaKC9MalDjUW9iN6Jl/OocTDtXdeAj7pEGIRh4=",
|
||||||
"owner": "NixOS",
|
"owner": "NixOS",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "0cbe9f69c234a7700596e943bfae7ef27a31b735",
|
"rev": "4bc72cae107788bf3f24f30db2e2f685c9298dc9",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@ -153,11 +138,11 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1698852633,
|
"lastModified": 1689668210,
|
||||||
"narHash": "sha256-Hsc/cCHud8ZXLvmm8pxrXpuaPEeNaaUttaCvtdX/Wug=",
|
"narHash": "sha256-XAATwDkaUxH958yXLs1lcEOmU6pSEIkatY3qjqk8X0E=",
|
||||||
"owner": "cachix",
|
"owner": "cachix",
|
||||||
"repo": "pre-commit-hooks.nix",
|
"repo": "pre-commit-hooks.nix",
|
||||||
"rev": "dec10399e5b56aa95fcd530e0338be72ad6462a0",
|
"rev": "eb433bff05b285258be76513add6f6c57b441775",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@ -171,7 +156,6 @@
|
|||||||
"flake-compat": "flake-compat",
|
"flake-compat": "flake-compat",
|
||||||
"flake-parts": "flake-parts",
|
"flake-parts": "flake-parts",
|
||||||
"libnbtplusplus": "libnbtplusplus",
|
"libnbtplusplus": "libnbtplusplus",
|
||||||
"nix-filter": "nix-filter",
|
|
||||||
"nixpkgs": "nixpkgs",
|
"nixpkgs": "nixpkgs",
|
||||||
"pre-commit-hooks": "pre-commit-hooks"
|
"pre-commit-hooks": "pre-commit-hooks"
|
||||||
}
|
}
|
||||||
|
25
flake.nix
25
flake.nix
@ -4,7 +4,6 @@
|
|||||||
inputs = {
|
inputs = {
|
||||||
nixpkgs.url = "github:nixos/nixpkgs/nixpkgs-unstable";
|
nixpkgs.url = "github:nixos/nixpkgs/nixpkgs-unstable";
|
||||||
flake-parts.url = "github:hercules-ci/flake-parts";
|
flake-parts.url = "github:hercules-ci/flake-parts";
|
||||||
nix-filter.url = "github:numtide/nix-filter";
|
|
||||||
pre-commit-hooks = {
|
pre-commit-hooks = {
|
||||||
url = "github:cachix/pre-commit-hooks.nix";
|
url = "github:cachix/pre-commit-hooks.nix";
|
||||||
inputs.nixpkgs.follows = "nixpkgs";
|
inputs.nixpkgs.follows = "nixpkgs";
|
||||||
@ -21,24 +20,8 @@
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
outputs = {
|
outputs = inputs:
|
||||||
flake-parts,
|
inputs.flake-parts.lib.mkFlake
|
||||||
pre-commit-hooks,
|
{inherit inputs;}
|
||||||
...
|
{imports = [./nix];};
|
||||||
} @ inputs:
|
|
||||||
flake-parts.lib.mkFlake {inherit inputs;} {
|
|
||||||
imports = [
|
|
||||||
pre-commit-hooks.flakeModule
|
|
||||||
|
|
||||||
./nix/dev.nix
|
|
||||||
./nix/distribution.nix
|
|
||||||
];
|
|
||||||
|
|
||||||
systems = [
|
|
||||||
"x86_64-linux"
|
|
||||||
"aarch64-linux"
|
|
||||||
"x86_64-darwin"
|
|
||||||
"aarch64-darwin"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
@ -1,22 +1,22 @@
|
|||||||
{
|
{
|
||||||
"name": "libdecor",
|
"name": "libdecor",
|
||||||
"buildsystem": "meson",
|
"buildsystem": "meson",
|
||||||
"config-opts": [
|
"config-opts": [
|
||||||
"-Ddemo=false"
|
"-Ddemo=false"
|
||||||
],
|
],
|
||||||
"sources": [
|
"sources": [
|
||||||
{
|
{
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://gitlab.freedesktop.org/libdecor/libdecor.git",
|
"url": "https://gitlab.freedesktop.org/libdecor/libdecor.git",
|
||||||
"commit": "73260393a97291c887e1074ab7f318e031be0ac6"
|
"commit": "73260393a97291c887e1074ab7f318e031be0ac6"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"type": "patch",
|
"type": "patch",
|
||||||
"path": "patches/weird_libdecor.patch"
|
"path": "patches/weird_libdecor.patch"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"cleanup": [
|
"cleanup": [
|
||||||
"/include",
|
"/include",
|
||||||
"/lib/pkgconfig"
|
"/lib/pkgconfig"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
@ -18,8 +18,6 @@ finish-args:
|
|||||||
- --filesystem=xdg-run/app/com.discordapp.Discord:create
|
- --filesystem=xdg-run/app/com.discordapp.Discord:create
|
||||||
# Mod drag&drop
|
# Mod drag&drop
|
||||||
- --filesystem=xdg-download:ro
|
- --filesystem=xdg-download:ro
|
||||||
# FTBApp import
|
|
||||||
- --filesystem=~/.ftba:ro
|
|
||||||
|
|
||||||
cleanup:
|
cleanup:
|
||||||
- /lib/libGLU*
|
- /lib/libGLU*
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
builds:
|
builds:
|
||||||
exclude: []
|
exclude: []
|
||||||
include:
|
include:
|
||||||
- "checks.x86_64-linux.*"
|
- "devShells.*-linux.*"
|
||||||
- "devShells.*.*"
|
- "packages.*-linux.*"
|
||||||
- "packages.*.*"
|
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -2,7 +2,7 @@
|
|||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* Prism Launcher - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
* Copyright (C) 2022 Tayou <git@tayou.org>
|
* Copyright (C) 2022 Tayou <tayou@gmx.net>
|
||||||
* Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me>
|
* Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
@ -38,17 +38,16 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QApplication>
|
#include <QApplication>
|
||||||
#include <QDateTime>
|
#include <memory>
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
#include <QFlag>
|
#include <QFlag>
|
||||||
#include <QIcon>
|
#include <QIcon>
|
||||||
|
#include <QDateTime>
|
||||||
#include <QUrl>
|
#include <QUrl>
|
||||||
#include <memory>
|
|
||||||
|
|
||||||
#include <BaseInstance.h>
|
#include <BaseInstance.h>
|
||||||
|
|
||||||
#include "minecraft/launch/MinecraftServerTarget.h"
|
#include "minecraft/launch/MinecraftServerTarget.h"
|
||||||
#include "ui/themes/CatPack.h"
|
|
||||||
|
|
||||||
class LaunchController;
|
class LaunchController;
|
||||||
class LocalPeer;
|
class LocalPeer;
|
||||||
@ -71,22 +70,27 @@ class TranslationsModel;
|
|||||||
class ITheme;
|
class ITheme;
|
||||||
class MCEditTool;
|
class MCEditTool;
|
||||||
class ThemeManager;
|
class ThemeManager;
|
||||||
class IconTheme;
|
|
||||||
|
|
||||||
namespace Meta {
|
namespace Meta {
|
||||||
class Index;
|
class Index;
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(APPLICATION)
|
#if defined(APPLICATION)
|
||||||
#undef APPLICATION
|
#undef APPLICATION
|
||||||
#endif
|
#endif
|
||||||
#define APPLICATION (static_cast<Application*>(QCoreApplication::instance()))
|
#define APPLICATION (static_cast<Application *>(QCoreApplication::instance()))
|
||||||
|
|
||||||
class Application : public QApplication {
|
class Application : public QApplication
|
||||||
|
{
|
||||||
// friends for the purpose of limiting access to deprecated stuff
|
// friends for the purpose of limiting access to deprecated stuff
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
enum Status { StartingUp, Failed, Succeeded, Initialized };
|
enum Status {
|
||||||
|
StartingUp,
|
||||||
|
Failed,
|
||||||
|
Succeeded,
|
||||||
|
Initialized
|
||||||
|
};
|
||||||
|
|
||||||
enum Capability {
|
enum Capability {
|
||||||
None = 0,
|
None = 0,
|
||||||
@ -98,21 +102,33 @@ class Application : public QApplication {
|
|||||||
};
|
};
|
||||||
Q_DECLARE_FLAGS(Capabilities, Capability)
|
Q_DECLARE_FLAGS(Capabilities, Capability)
|
||||||
|
|
||||||
public:
|
public:
|
||||||
Application(int& argc, char** argv);
|
Application(int &argc, char **argv);
|
||||||
virtual ~Application();
|
virtual ~Application();
|
||||||
|
|
||||||
bool event(QEvent* event) override;
|
bool event(QEvent* event) override;
|
||||||
|
|
||||||
std::shared_ptr<SettingsObject> settings() const { return m_settings; }
|
std::shared_ptr<SettingsObject> settings() const {
|
||||||
|
return m_settings;
|
||||||
|
}
|
||||||
|
|
||||||
qint64 timeSinceStart() const { return startTime.msecsTo(QDateTime::currentDateTime()); }
|
qint64 timeSinceStart() const {
|
||||||
|
return startTime.msecsTo(QDateTime::currentDateTime());
|
||||||
|
}
|
||||||
|
|
||||||
QIcon getThemedIcon(const QString& name);
|
QIcon getThemedIcon(const QString& name);
|
||||||
|
|
||||||
ThemeManager* themeManager() { return m_themeManager.get(); }
|
void setIconTheme(const QString& name);
|
||||||
|
|
||||||
shared_qobject_ptr<ExternalUpdater> updater() { return m_updater; }
|
void applyCurrentlySelectedTheme(bool initial = false);
|
||||||
|
|
||||||
|
QList<ITheme*> getValidApplicationThemes();
|
||||||
|
|
||||||
|
void setApplicationTheme(const QString& name);
|
||||||
|
|
||||||
|
shared_qobject_ptr<ExternalUpdater> updater() {
|
||||||
|
return m_updater;
|
||||||
|
}
|
||||||
|
|
||||||
void triggerUpdateCheck();
|
void triggerUpdateCheck();
|
||||||
|
|
||||||
@ -120,17 +136,29 @@ class Application : public QApplication {
|
|||||||
|
|
||||||
std::shared_ptr<JavaInstallList> javalist();
|
std::shared_ptr<JavaInstallList> javalist();
|
||||||
|
|
||||||
std::shared_ptr<InstanceList> instances() const { return m_instances; }
|
std::shared_ptr<InstanceList> instances() const {
|
||||||
|
return m_instances;
|
||||||
|
}
|
||||||
|
|
||||||
std::shared_ptr<IconList> icons() const { return m_icons; }
|
std::shared_ptr<IconList> icons() const {
|
||||||
|
return m_icons;
|
||||||
|
}
|
||||||
|
|
||||||
MCEditTool* mcedit() const { return m_mcedit.get(); }
|
MCEditTool *mcedit() const {
|
||||||
|
return m_mcedit.get();
|
||||||
|
}
|
||||||
|
|
||||||
shared_qobject_ptr<AccountList> accounts() const { return m_accounts; }
|
shared_qobject_ptr<AccountList> accounts() const {
|
||||||
|
return m_accounts;
|
||||||
|
}
|
||||||
|
|
||||||
Status status() const { return m_status; }
|
Status status() const {
|
||||||
|
return m_status;
|
||||||
|
}
|
||||||
|
|
||||||
const QMap<QString, std::shared_ptr<BaseProfilerFactory>>& profilers() const { return m_profilers; }
|
const QMap<QString, std::shared_ptr<BaseProfilerFactory>> &profilers() const {
|
||||||
|
return m_profilers;
|
||||||
|
}
|
||||||
|
|
||||||
void updateProxySettings(QString proxyTypeStr, QString addr, int port, QString user, QString password);
|
void updateProxySettings(QString proxyTypeStr, QString addr, int port, QString user, QString password);
|
||||||
|
|
||||||
@ -142,8 +170,6 @@ class Application : public QApplication {
|
|||||||
|
|
||||||
void updateCapabilities();
|
void updateCapabilities();
|
||||||
|
|
||||||
void detectLibraries();
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Finds and returns the full path to a jar file.
|
* Finds and returns the full path to a jar file.
|
||||||
* Returns a null-string if it could not be found.
|
* Returns a null-string if it could not be found.
|
||||||
@ -157,37 +183,35 @@ class Application : public QApplication {
|
|||||||
QString getUserAgentUncached();
|
QString getUserAgentUncached();
|
||||||
|
|
||||||
/// this is the root of the 'installation'. Used for automatic updates
|
/// this is the root of the 'installation'. Used for automatic updates
|
||||||
const QString& root() { return m_rootPath; }
|
const QString &root() {
|
||||||
|
return m_rootPath;
|
||||||
|
}
|
||||||
|
|
||||||
/// the data path the application is using
|
bool isPortable() {
|
||||||
const QString& dataRoot() { return m_dataPath; }
|
return m_portable;
|
||||||
|
}
|
||||||
|
|
||||||
bool isPortable() { return m_portable; }
|
const Capabilities capabilities() {
|
||||||
|
return m_capabilities;
|
||||||
const Capabilities capabilities() { return m_capabilities; }
|
}
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Opens a json file using either a system default editor, or, if not empty, the editor
|
* Opens a json file using either a system default editor, or, if not empty, the editor
|
||||||
* specified in the settings
|
* specified in the settings
|
||||||
*/
|
*/
|
||||||
bool openJsonEditor(const QString& filename);
|
bool openJsonEditor(const QString &filename);
|
||||||
|
|
||||||
InstanceWindow* showInstanceWindow(InstancePtr instance, QString page = QString());
|
InstanceWindow *showInstanceWindow(InstancePtr instance, QString page = QString());
|
||||||
MainWindow* showMainWindow(bool minimized = false);
|
MainWindow *showMainWindow(bool minimized = false);
|
||||||
|
|
||||||
void updateIsRunning(bool running);
|
void updateIsRunning(bool running);
|
||||||
bool updatesAreAllowed();
|
bool updatesAreAllowed();
|
||||||
|
|
||||||
void ShowGlobalSettings(class QWidget* parent, QString open_page = QString());
|
void ShowGlobalSettings(class QWidget * parent, QString open_page = QString());
|
||||||
|
|
||||||
int suitableMaxMem();
|
int suitableMaxMem();
|
||||||
|
|
||||||
bool updaterEnabled();
|
signals:
|
||||||
QString updaterBinaryName();
|
|
||||||
|
|
||||||
QUrl normalizeImportUrl(QString const& url);
|
|
||||||
|
|
||||||
signals:
|
|
||||||
void updateAllowedChanged(bool status);
|
void updateAllowedChanged(bool status);
|
||||||
void globalSettingsAboutToOpen();
|
void globalSettingsAboutToOpen();
|
||||||
void globalSettingsClosed();
|
void globalSettingsClosed();
|
||||||
@ -197,36 +221,39 @@ class Application : public QApplication {
|
|||||||
void clickedOnDock();
|
void clickedOnDock();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
bool launch(InstancePtr instance,
|
bool launch(
|
||||||
bool online = true,
|
InstancePtr instance,
|
||||||
bool demo = false,
|
bool online = true,
|
||||||
MinecraftServerTargetPtr serverToJoin = nullptr,
|
bool demo = false,
|
||||||
MinecraftAccountPtr accountToUse = nullptr);
|
BaseProfilerFactory *profiler = nullptr,
|
||||||
|
MinecraftServerTargetPtr serverToJoin = nullptr,
|
||||||
|
MinecraftAccountPtr accountToUse = nullptr
|
||||||
|
);
|
||||||
bool kill(InstancePtr instance);
|
bool kill(InstancePtr instance);
|
||||||
void closeCurrentWindow();
|
void closeCurrentWindow();
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void on_windowClose();
|
void on_windowClose();
|
||||||
void messageReceived(const QByteArray& message);
|
void messageReceived(const QByteArray & message);
|
||||||
void controllerSucceeded();
|
void controllerSucceeded();
|
||||||
void controllerFailed(const QString& error);
|
void controllerFailed(const QString & error);
|
||||||
void setupWizardFinished(int status);
|
void setupWizardFinished(int status);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool handleDataMigration(const QString& currentData, const QString& oldData, const QString& name, const QString& configFile) const;
|
bool handleDataMigration(const QString & currentData, const QString & oldData, const QString & name, const QString & configFile) const;
|
||||||
bool createSetupWizard();
|
bool createSetupWizard();
|
||||||
void performMainStartupAction();
|
void performMainStartupAction();
|
||||||
|
|
||||||
// sets the fatal error message and m_status to Failed.
|
// sets the fatal error message and m_status to Failed.
|
||||||
void showFatalErrorMessage(const QString& title, const QString& content);
|
void showFatalErrorMessage(const QString & title, const QString & content);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void addRunningInstance();
|
void addRunningInstance();
|
||||||
void subRunningInstance();
|
void subRunningInstance();
|
||||||
bool shouldExitNow() const;
|
bool shouldExitNow() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QDateTime startTime;
|
QDateTime startTime;
|
||||||
|
|
||||||
shared_qobject_ptr<QNetworkAccessManager> m_network;
|
shared_qobject_ptr<QNetworkAccessManager> m_network;
|
||||||
@ -250,10 +277,9 @@ class Application : public QApplication {
|
|||||||
QMap<QString, std::shared_ptr<BaseProfilerFactory>> m_profilers;
|
QMap<QString, std::shared_ptr<BaseProfilerFactory>> m_profilers;
|
||||||
|
|
||||||
QString m_rootPath;
|
QString m_rootPath;
|
||||||
QString m_dataPath;
|
|
||||||
Status m_status = Application::StartingUp;
|
Status m_status = Application::StartingUp;
|
||||||
Capabilities m_capabilities;
|
Capabilities m_capabilities;
|
||||||
bool m_portable = false;
|
bool m_portable = false;
|
||||||
|
|
||||||
#ifdef Q_OS_MACOS
|
#ifdef Q_OS_MACOS
|
||||||
Qt::ApplicationState m_prevAppState = Qt::ApplicationInactive;
|
Qt::ApplicationState m_prevAppState = Qt::ApplicationInactive;
|
||||||
@ -266,7 +292,7 @@ class Application : public QApplication {
|
|||||||
|
|
||||||
// FIXME: attach to instances instead.
|
// FIXME: attach to instances instead.
|
||||||
struct InstanceXtras {
|
struct InstanceXtras {
|
||||||
InstanceWindow* window = nullptr;
|
InstanceWindow * window = nullptr;
|
||||||
shared_qobject_ptr<LaunchController> controller;
|
shared_qobject_ptr<LaunchController> controller;
|
||||||
};
|
};
|
||||||
std::map<QString, InstanceXtras> m_instanceExtras;
|
std::map<QString, InstanceXtras> m_instanceExtras;
|
||||||
@ -277,21 +303,18 @@ class Application : public QApplication {
|
|||||||
bool m_updateRunning = false;
|
bool m_updateRunning = false;
|
||||||
|
|
||||||
// main window, if any
|
// main window, if any
|
||||||
MainWindow* m_mainWindow = nullptr;
|
MainWindow * m_mainWindow = nullptr;
|
||||||
|
|
||||||
// peer launcher instance connector - used to implement single instance launcher and signalling
|
// peer launcher instance connector - used to implement single instance launcher and signalling
|
||||||
LocalPeer* m_peerInstance = nullptr;
|
LocalPeer * m_peerInstance = nullptr;
|
||||||
|
|
||||||
SetupWizard* m_setupWizard = nullptr;
|
SetupWizard * m_setupWizard = nullptr;
|
||||||
|
public:
|
||||||
public:
|
|
||||||
QString m_detectedGLFWPath;
|
|
||||||
QString m_detectedOpenALPath;
|
|
||||||
QString m_instanceIdToLaunch;
|
QString m_instanceIdToLaunch;
|
||||||
QString m_serverToJoin;
|
QString m_serverToJoin;
|
||||||
QString m_profileToUse;
|
QString m_profileToUse;
|
||||||
bool m_liveCheck = false;
|
bool m_liveCheck = false;
|
||||||
QList<QUrl> m_urlsToImport;
|
QList<QUrl> m_zipsToImport;
|
||||||
QString m_instanceIdToShowWindowOf;
|
QString m_instanceIdToShowWindowOf;
|
||||||
std::unique_ptr<QFile> logFile;
|
std::unique_ptr<QFile> logFile;
|
||||||
};
|
};
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
@ -39,8 +39,7 @@
|
|||||||
#include <QJsonObject>
|
#include <QJsonObject>
|
||||||
#include "Json.h"
|
#include "Json.h"
|
||||||
|
|
||||||
void ApplicationMessage::parse(const QByteArray& input)
|
void ApplicationMessage::parse(const QByteArray & input) {
|
||||||
{
|
|
||||||
auto doc = Json::requireDocument(input, "ApplicationMessage");
|
auto doc = Json::requireDocument(input, "ApplicationMessage");
|
||||||
auto root = Json::requireObject(doc, "ApplicationMessage");
|
auto root = Json::requireObject(doc, "ApplicationMessage");
|
||||||
|
|
||||||
@ -48,13 +47,12 @@ void ApplicationMessage::parse(const QByteArray& input)
|
|||||||
args.clear();
|
args.clear();
|
||||||
|
|
||||||
auto parsedArgs = root.value("args").toObject();
|
auto parsedArgs = root.value("args").toObject();
|
||||||
for (auto iter = parsedArgs.constBegin(); iter != parsedArgs.constEnd(); iter++) {
|
for(auto iter = parsedArgs.constBegin(); iter != parsedArgs.constEnd(); iter++) {
|
||||||
args.insert(iter.key(), iter.value().toString());
|
args.insert(iter.key(), iter.value().toString());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
QByteArray ApplicationMessage::serialize()
|
QByteArray ApplicationMessage::serialize() {
|
||||||
{
|
|
||||||
QJsonObject root;
|
QJsonObject root;
|
||||||
root.insert("command", command);
|
root.insert("command", command);
|
||||||
QJsonObject outArgs;
|
QJsonObject outArgs;
|
||||||
|
@ -1,13 +1,13 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QByteArray>
|
|
||||||
#include <QHash>
|
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
#include <QHash>
|
||||||
|
#include <QByteArray>
|
||||||
|
|
||||||
struct ApplicationMessage {
|
struct ApplicationMessage {
|
||||||
QString command;
|
QString command;
|
||||||
QHash<QString, QString> args;
|
QHash<QString, QString> args;
|
||||||
|
|
||||||
QByteArray serialize();
|
QByteArray serialize();
|
||||||
void parse(const QByteArray& input);
|
void parse(const QByteArray & input);
|
||||||
};
|
};
|
||||||
|
@ -18,21 +18,27 @@
|
|||||||
#include "BaseInstaller.h"
|
#include "BaseInstaller.h"
|
||||||
#include "minecraft/MinecraftInstance.h"
|
#include "minecraft/MinecraftInstance.h"
|
||||||
|
|
||||||
BaseInstaller::BaseInstaller() {}
|
BaseInstaller::BaseInstaller()
|
||||||
|
{
|
||||||
|
|
||||||
bool BaseInstaller::isApplied(MinecraftInstance* on)
|
}
|
||||||
|
|
||||||
|
bool BaseInstaller::isApplied(MinecraftInstance *on)
|
||||||
{
|
{
|
||||||
return QFile::exists(filename(on->instanceRoot()));
|
return QFile::exists(filename(on->instanceRoot()));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool BaseInstaller::add(MinecraftInstance* to)
|
bool BaseInstaller::add(MinecraftInstance *to)
|
||||||
{
|
{
|
||||||
if (!patchesDir(to->instanceRoot()).exists()) {
|
if (!patchesDir(to->instanceRoot()).exists())
|
||||||
|
{
|
||||||
QDir(to->instanceRoot()).mkdir("patches");
|
QDir(to->instanceRoot()).mkdir("patches");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isApplied(to)) {
|
if (isApplied(to))
|
||||||
if (!remove(to)) {
|
{
|
||||||
|
if (!remove(to))
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -40,16 +46,16 @@ bool BaseInstaller::add(MinecraftInstance* to)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool BaseInstaller::remove(MinecraftInstance* from)
|
bool BaseInstaller::remove(MinecraftInstance *from)
|
||||||
{
|
{
|
||||||
return QFile::remove(filename(from->instanceRoot()));
|
return QFile::remove(filename(from->instanceRoot()));
|
||||||
}
|
}
|
||||||
|
|
||||||
QString BaseInstaller::filename(const QString& root) const
|
QString BaseInstaller::filename(const QString &root) const
|
||||||
{
|
{
|
||||||
return patchesDir(root).absoluteFilePath(id() + ".json");
|
return patchesDir(root).absoluteFilePath(id() + ".json");
|
||||||
}
|
}
|
||||||
QDir BaseInstaller::patchesDir(const QString& root) const
|
QDir BaseInstaller::patchesDir(const QString &root) const
|
||||||
{
|
{
|
||||||
return QDir(root + "/patches/");
|
return QDir(root + "/patches/");
|
||||||
}
|
}
|
||||||
|
@ -26,19 +26,20 @@ class QObject;
|
|||||||
class Task;
|
class Task;
|
||||||
class BaseVersion;
|
class BaseVersion;
|
||||||
|
|
||||||
class BaseInstaller {
|
class BaseInstaller
|
||||||
public:
|
{
|
||||||
|
public:
|
||||||
BaseInstaller();
|
BaseInstaller();
|
||||||
virtual ~BaseInstaller(){};
|
virtual ~BaseInstaller(){};
|
||||||
bool isApplied(MinecraftInstance* on);
|
bool isApplied(MinecraftInstance *on);
|
||||||
|
|
||||||
virtual bool add(MinecraftInstance* to);
|
virtual bool add(MinecraftInstance *to);
|
||||||
virtual bool remove(MinecraftInstance* from);
|
virtual bool remove(MinecraftInstance *from);
|
||||||
|
|
||||||
virtual Task* createInstallTask(MinecraftInstance* instance, BaseVersion::Ptr version, QObject* parent) = 0;
|
virtual Task *createInstallTask(MinecraftInstance *instance, BaseVersion::Ptr version, QObject *parent) = 0;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
virtual QString id() const = 0;
|
virtual QString id() const = 0;
|
||||||
QString filename(const QString& root) const;
|
QString filename(const QString &root) const;
|
||||||
QDir patchesDir(const QString& root) const;
|
QDir patchesDir(const QString &root) const;
|
||||||
};
|
};
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
* Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org>
|
* Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org>
|
||||||
* Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me>
|
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -37,22 +36,23 @@
|
|||||||
|
|
||||||
#include "BaseInstance.h"
|
#include "BaseInstance.h"
|
||||||
|
|
||||||
#include <QDebug>
|
|
||||||
#include <QDir>
|
|
||||||
#include <QFileInfo>
|
#include <QFileInfo>
|
||||||
|
#include <QDir>
|
||||||
|
#include <QDebug>
|
||||||
|
#include <QRegularExpression>
|
||||||
#include <QJsonDocument>
|
#include <QJsonDocument>
|
||||||
#include <QJsonObject>
|
#include <QJsonObject>
|
||||||
#include <QRegularExpression>
|
|
||||||
|
|
||||||
#include "settings/INISettingsObject.h"
|
#include "settings/INISettingsObject.h"
|
||||||
#include "settings/OverrideSetting.h"
|
|
||||||
#include "settings/Setting.h"
|
#include "settings/Setting.h"
|
||||||
|
#include "settings/OverrideSetting.h"
|
||||||
|
|
||||||
#include "BuildConfig.h"
|
|
||||||
#include "Commandline.h"
|
|
||||||
#include "FileSystem.h"
|
#include "FileSystem.h"
|
||||||
|
#include "Commandline.h"
|
||||||
|
#include "BuildConfig.h"
|
||||||
|
|
||||||
BaseInstance::BaseInstance(SettingsObjectPtr globalSettings, SettingsObjectPtr settings, const QString& rootDir) : QObject()
|
BaseInstance::BaseInstance(SettingsObjectPtr globalSettings, SettingsObjectPtr settings, const QString &rootDir)
|
||||||
|
: QObject()
|
||||||
{
|
{
|
||||||
m_settings = settings;
|
m_settings = settings;
|
||||||
m_global_settings = globalSettings;
|
m_global_settings = globalSettings;
|
||||||
@ -79,7 +79,7 @@ BaseInstance::BaseInstance(SettingsObjectPtr globalSettings, SettingsObjectPtr s
|
|||||||
m_settings->registerSetting("InstanceType", "");
|
m_settings->registerSetting("InstanceType", "");
|
||||||
|
|
||||||
// Custom Commands
|
// Custom Commands
|
||||||
auto commandSetting = m_settings->registerSetting({ "OverrideCommands", "OverrideLaunchCmd" }, false);
|
auto commandSetting = m_settings->registerSetting({"OverrideCommands","OverrideLaunchCmd"}, false);
|
||||||
m_settings->registerOverride(globalSettings->getSetting("PreLaunchCommand"), commandSetting);
|
m_settings->registerOverride(globalSettings->getSetting("PreLaunchCommand"), commandSetting);
|
||||||
m_settings->registerOverride(globalSettings->getSetting("WrapperCommand"), commandSetting);
|
m_settings->registerOverride(globalSettings->getSetting("WrapperCommand"), commandSetting);
|
||||||
m_settings->registerOverride(globalSettings->getSetting("PostExitCommand"), commandSetting);
|
m_settings->registerOverride(globalSettings->getSetting("PostExitCommand"), commandSetting);
|
||||||
@ -101,8 +101,6 @@ BaseInstance::BaseInstance(SettingsObjectPtr globalSettings, SettingsObjectPtr s
|
|||||||
m_settings->registerSetting("ManagedPackName", "");
|
m_settings->registerSetting("ManagedPackName", "");
|
||||||
m_settings->registerSetting("ManagedPackVersionID", "");
|
m_settings->registerSetting("ManagedPackVersionID", "");
|
||||||
m_settings->registerSetting("ManagedPackVersionName", "");
|
m_settings->registerSetting("ManagedPackVersionName", "");
|
||||||
|
|
||||||
m_settings->registerSetting("Profiler", "");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
QString BaseInstance::getPreLaunchCommand()
|
QString BaseInstance::getPreLaunchCommand()
|
||||||
@ -150,11 +148,7 @@ QString BaseInstance::getManagedPackVersionName() const
|
|||||||
return m_settings->get("ManagedPackVersionName").toString();
|
return m_settings->get("ManagedPackVersionName").toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
void BaseInstance::setManagedPack(const QString& type,
|
void BaseInstance::setManagedPack(const QString& type, const QString& id, const QString& name, const QString& versionId, const QString& version)
|
||||||
const QString& id,
|
|
||||||
const QString& name,
|
|
||||||
const QString& versionId,
|
|
||||||
const QString& version)
|
|
||||||
{
|
{
|
||||||
m_settings->set("ManagedPack", true);
|
m_settings->set("ManagedPack", true);
|
||||||
m_settings->set("ManagedPackType", type);
|
m_settings->set("ManagedPackType", type);
|
||||||
@ -179,7 +173,8 @@ int BaseInstance::getConsoleMaxLines() const
|
|||||||
auto lineSetting = m_settings->getSetting("ConsoleMaxLines");
|
auto lineSetting = m_settings->getSetting("ConsoleMaxLines");
|
||||||
bool conversionOk = false;
|
bool conversionOk = false;
|
||||||
int maxLines = lineSetting->get().toInt(&conversionOk);
|
int maxLines = lineSetting->get().toInt(&conversionOk);
|
||||||
if (!conversionOk) {
|
if(!conversionOk)
|
||||||
|
{
|
||||||
maxLines = lineSetting->defValue().toInt();
|
maxLines = lineSetting->defValue().toInt();
|
||||||
qWarning() << "ConsoleMaxLines has nonsensical value, defaulting to" << maxLines;
|
qWarning() << "ConsoleMaxLines has nonsensical value, defaulting to" << maxLines;
|
||||||
}
|
}
|
||||||
@ -225,7 +220,8 @@ bool BaseInstance::isLinkedToInstanceId(const QString& id) const
|
|||||||
|
|
||||||
void BaseInstance::iconUpdated(QString key)
|
void BaseInstance::iconUpdated(QString key)
|
||||||
{
|
{
|
||||||
if (iconKey() == key) {
|
if(iconKey() == key)
|
||||||
|
{
|
||||||
emit propertiesChanged(this);
|
emit propertiesChanged(this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -239,7 +235,8 @@ void BaseInstance::invalidate()
|
|||||||
void BaseInstance::changeStatus(BaseInstance::Status newStatus)
|
void BaseInstance::changeStatus(BaseInstance::Status newStatus)
|
||||||
{
|
{
|
||||||
Status status = currentStatus();
|
Status status = currentStatus();
|
||||||
if (status != newStatus) {
|
if(status != newStatus)
|
||||||
|
{
|
||||||
m_status = newStatus;
|
m_status = newStatus;
|
||||||
emit statusChanged(status, newStatus);
|
emit statusChanged(status, newStatus);
|
||||||
}
|
}
|
||||||
@ -262,19 +259,23 @@ bool BaseInstance::isRunning() const
|
|||||||
|
|
||||||
void BaseInstance::setRunning(bool running)
|
void BaseInstance::setRunning(bool running)
|
||||||
{
|
{
|
||||||
if (running == m_isRunning)
|
if(running == m_isRunning)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
m_isRunning = running;
|
m_isRunning = running;
|
||||||
|
|
||||||
if (!m_settings->get("RecordGameTime").toBool()) {
|
if(!m_settings->get("RecordGameTime").toBool())
|
||||||
|
{
|
||||||
emit runningStatusChanged(running);
|
emit runningStatusChanged(running);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (running) {
|
if(running)
|
||||||
|
{
|
||||||
m_timeStarted = QDateTime::currentDateTime();
|
m_timeStarted = QDateTime::currentDateTime();
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
QDateTime timeEnded = QDateTime::currentDateTime();
|
QDateTime timeEnded = QDateTime::currentDateTime();
|
||||||
|
|
||||||
qint64 current = settings()->get("totalTimePlayed").toLongLong();
|
qint64 current = settings()->get("totalTimePlayed").toLongLong();
|
||||||
@ -290,7 +291,8 @@ void BaseInstance::setRunning(bool running)
|
|||||||
int64_t BaseInstance::totalTimePlayed() const
|
int64_t BaseInstance::totalTimePlayed() const
|
||||||
{
|
{
|
||||||
qint64 current = m_settings->get("totalTimePlayed").toLongLong();
|
qint64 current = m_settings->get("totalTimePlayed").toLongLong();
|
||||||
if (m_isRunning) {
|
if(m_isRunning)
|
||||||
|
{
|
||||||
QDateTime timeNow = QDateTime::currentDateTime();
|
QDateTime timeNow = QDateTime::currentDateTime();
|
||||||
return current + m_timeStarted.secsTo(timeNow);
|
return current + m_timeStarted.secsTo(timeNow);
|
||||||
}
|
}
|
||||||
@ -299,7 +301,8 @@ int64_t BaseInstance::totalTimePlayed() const
|
|||||||
|
|
||||||
int64_t BaseInstance::lastTimePlayed() const
|
int64_t BaseInstance::lastTimePlayed() const
|
||||||
{
|
{
|
||||||
if (m_isRunning) {
|
if(m_isRunning)
|
||||||
|
{
|
||||||
QDateTime timeNow = QDateTime::currentDateTime();
|
QDateTime timeNow = QDateTime::currentDateTime();
|
||||||
return m_timeStarted.secsTo(timeNow);
|
return m_timeStarted.secsTo(timeNow);
|
||||||
}
|
}
|
||||||
@ -346,14 +349,14 @@ qint64 BaseInstance::lastLaunch() const
|
|||||||
|
|
||||||
void BaseInstance::setLastLaunch(qint64 val)
|
void BaseInstance::setLastLaunch(qint64 val)
|
||||||
{
|
{
|
||||||
// FIXME: if no change, do not set. setting involves saving a file.
|
//FIXME: if no change, do not set. setting involves saving a file.
|
||||||
m_settings->set("lastLaunchTime", val);
|
m_settings->set("lastLaunchTime", val);
|
||||||
emit propertiesChanged(this);
|
emit propertiesChanged(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
void BaseInstance::setNotes(QString val)
|
void BaseInstance::setNotes(QString val)
|
||||||
{
|
{
|
||||||
// FIXME: if no change, do not set. setting involves saving a file.
|
//FIXME: if no change, do not set. setting involves saving a file.
|
||||||
m_settings->set("notes", val);
|
m_settings->set("notes", val);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -364,7 +367,7 @@ QString BaseInstance::notes() const
|
|||||||
|
|
||||||
void BaseInstance::setIconKey(QString val)
|
void BaseInstance::setIconKey(QString val)
|
||||||
{
|
{
|
||||||
// FIXME: if no change, do not set. setting involves saving a file.
|
//FIXME: if no change, do not set. setting involves saving a file.
|
||||||
m_settings->set("iconKey", val);
|
m_settings->set("iconKey", val);
|
||||||
emit propertiesChanged(this);
|
emit propertiesChanged(this);
|
||||||
}
|
}
|
||||||
@ -376,7 +379,7 @@ QString BaseInstance::iconKey() const
|
|||||||
|
|
||||||
void BaseInstance::setName(QString val)
|
void BaseInstance::setName(QString val)
|
||||||
{
|
{
|
||||||
// FIXME: if no change, do not set. setting involves saving a file.
|
//FIXME: if no change, do not set. setting involves saving a file.
|
||||||
m_settings->set("name", val);
|
m_settings->set("name", val);
|
||||||
emit propertiesChanged(this);
|
emit propertiesChanged(this);
|
||||||
}
|
}
|
||||||
@ -388,7 +391,7 @@ QString BaseInstance::name() const
|
|||||||
|
|
||||||
QString BaseInstance::windowTitle() const
|
QString BaseInstance::windowTitle() const
|
||||||
{
|
{
|
||||||
return BuildConfig.LAUNCHER_DISPLAYNAME + ": " + name();
|
return BuildConfig.LAUNCHER_DISPLAYNAME + ": " + name().replace(QRegularExpression("\\s+"), " ");
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: why is this here? move it to MinecraftInstance!!!
|
// FIXME: why is this here? move it to MinecraftInstance!!!
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
* Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org>
|
* Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org>
|
||||||
* Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me>
|
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -38,25 +37,24 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
|
|
||||||
#include <QDateTime>
|
|
||||||
#include <QMenu>
|
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QProcess>
|
|
||||||
#include <QSet>
|
|
||||||
#include "QObjectPtr.h"
|
#include "QObjectPtr.h"
|
||||||
|
#include <QDateTime>
|
||||||
|
#include <QSet>
|
||||||
|
#include <QProcess>
|
||||||
|
|
||||||
#include "settings/SettingsObject.h"
|
#include "settings/SettingsObject.h"
|
||||||
|
|
||||||
#include "BaseVersionList.h"
|
|
||||||
#include "MessageLevel.h"
|
|
||||||
#include "minecraft/auth/MinecraftAccount.h"
|
|
||||||
#include "pathmatcher/IPathMatcher.h"
|
|
||||||
#include "settings/INIFile.h"
|
#include "settings/INIFile.h"
|
||||||
|
#include "BaseVersionList.h"
|
||||||
|
#include "minecraft/auth/MinecraftAccount.h"
|
||||||
|
#include "MessageLevel.h"
|
||||||
|
#include "pathmatcher/IPathMatcher.h"
|
||||||
|
|
||||||
#include "net/Mode.h"
|
#include "net/Mode.h"
|
||||||
|
|
||||||
#include "RuntimeContext.h"
|
|
||||||
#include "minecraft/launch/MinecraftServerTarget.h"
|
#include "minecraft/launch/MinecraftServerTarget.h"
|
||||||
|
#include "RuntimeContext.h"
|
||||||
|
|
||||||
class QDir;
|
class QDir;
|
||||||
class Task;
|
class Task;
|
||||||
@ -64,7 +62,7 @@ class LaunchTask;
|
|||||||
class BaseInstance;
|
class BaseInstance;
|
||||||
|
|
||||||
// pointer for lazy people
|
// pointer for lazy people
|
||||||
using InstancePtr = std::shared_ptr<BaseInstance>;
|
typedef std::shared_ptr<BaseInstance> InstancePtr;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \brief Base class for instances.
|
* \brief Base class for instances.
|
||||||
@ -74,21 +72,23 @@ using InstancePtr = std::shared_ptr<BaseInstance>;
|
|||||||
* To create a new instance type, create a new class inheriting from this class
|
* To create a new instance type, create a new class inheriting from this class
|
||||||
* and implement the pure virtual functions.
|
* and implement the pure virtual functions.
|
||||||
*/
|
*/
|
||||||
class BaseInstance : public QObject, public std::enable_shared_from_this<BaseInstance> {
|
class BaseInstance : public QObject, public std::enable_shared_from_this<BaseInstance>
|
||||||
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
protected:
|
protected:
|
||||||
/// no-touchy!
|
/// no-touchy!
|
||||||
BaseInstance(SettingsObjectPtr globalSettings, SettingsObjectPtr settings, const QString& rootDir);
|
BaseInstance(SettingsObjectPtr globalSettings, SettingsObjectPtr settings, const QString &rootDir);
|
||||||
|
|
||||||
public: /* types */
|
public: /* types */
|
||||||
enum class Status {
|
enum class Status
|
||||||
|
{
|
||||||
Present,
|
Present,
|
||||||
Gone // either nuked or invalidated
|
Gone // either nuked or invalidated
|
||||||
};
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/// virtual destructor to make sure the destruction is COMPLETE
|
/// virtual destructor to make sure the destruction is COMPLETE
|
||||||
virtual ~BaseInstance() {}
|
virtual ~BaseInstance() {};
|
||||||
|
|
||||||
virtual void saveNow() = 0;
|
virtual void saveNow() = 0;
|
||||||
|
|
||||||
@ -117,7 +117,10 @@ class BaseInstance : public QObject, public std::enable_shared_from_this<BaseIns
|
|||||||
QString instanceRoot() const;
|
QString instanceRoot() const;
|
||||||
|
|
||||||
/// Path to the instance's game root directory.
|
/// Path to the instance's game root directory.
|
||||||
virtual QString gameRoot() const { return instanceRoot(); }
|
virtual QString gameRoot() const
|
||||||
|
{
|
||||||
|
return instanceRoot();
|
||||||
|
}
|
||||||
|
|
||||||
/// Path to the instance's mods directory.
|
/// Path to the instance's mods directory.
|
||||||
virtual QString modsRoot() const = 0;
|
virtual QString modsRoot() const = 0;
|
||||||
@ -148,12 +151,15 @@ class BaseInstance : public QObject, public std::enable_shared_from_this<BaseIns
|
|||||||
void copyManagedPack(BaseInstance& other);
|
void copyManagedPack(BaseInstance& other);
|
||||||
|
|
||||||
/// guess log level from a line of game log
|
/// guess log level from a line of game log
|
||||||
virtual MessageLevel::Enum guessLevel([[maybe_unused]] const QString& line, MessageLevel::Enum level) { return level; }
|
virtual MessageLevel::Enum guessLevel([[maybe_unused]] const QString &line, MessageLevel::Enum level)
|
||||||
|
{
|
||||||
|
return level;
|
||||||
|
};
|
||||||
|
|
||||||
virtual QStringList extraArguments();
|
virtual QStringList extraArguments();
|
||||||
|
|
||||||
/// Traits. Normally inside the version, depends on instance implementation.
|
/// Traits. Normally inside the version, depends on instance implementation.
|
||||||
virtual QSet<QString> traits() const = 0;
|
virtual QSet <QString> traits() const = 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets the time that the instance was last launched.
|
* Gets the time that the instance was last launched.
|
||||||
@ -183,7 +189,8 @@ class BaseInstance : public QObject, public std::enable_shared_from_this<BaseIns
|
|||||||
virtual Task::Ptr createUpdateTask(Net::Mode mode) = 0;
|
virtual Task::Ptr createUpdateTask(Net::Mode mode) = 0;
|
||||||
|
|
||||||
/// returns a valid launcher (task container)
|
/// returns a valid launcher (task container)
|
||||||
virtual shared_qobject_ptr<LaunchTask> createLaunchTask(AuthSessionPtr account, MinecraftServerTargetPtr serverToJoin) = 0;
|
virtual shared_qobject_ptr<LaunchTask> createLaunchTask(
|
||||||
|
AuthSessionPtr account, MinecraftServerTargetPtr serverToJoin) = 0;
|
||||||
|
|
||||||
/// returns the current launch task (if any)
|
/// returns the current launch task (if any)
|
||||||
shared_qobject_ptr<LaunchTask> getLaunchTask();
|
shared_qobject_ptr<LaunchTask> getLaunchTask();
|
||||||
@ -215,30 +222,45 @@ class BaseInstance : public QObject, public std::enable_shared_from_this<BaseIns
|
|||||||
virtual QString typeName() const = 0;
|
virtual QString typeName() const = 0;
|
||||||
|
|
||||||
void updateRuntimeContext();
|
void updateRuntimeContext();
|
||||||
RuntimeContext runtimeContext() const { return m_runtimeContext; }
|
RuntimeContext runtimeContext() const
|
||||||
|
{
|
||||||
|
return m_runtimeContext;
|
||||||
|
}
|
||||||
|
|
||||||
bool hasVersionBroken() const { return m_hasBrokenVersion; }
|
bool hasVersionBroken() const
|
||||||
|
{
|
||||||
|
return m_hasBrokenVersion;
|
||||||
|
}
|
||||||
void setVersionBroken(bool value)
|
void setVersionBroken(bool value)
|
||||||
{
|
{
|
||||||
if (m_hasBrokenVersion != value) {
|
if(m_hasBrokenVersion != value)
|
||||||
|
{
|
||||||
m_hasBrokenVersion = value;
|
m_hasBrokenVersion = value;
|
||||||
emit propertiesChanged(this);
|
emit propertiesChanged(this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool hasUpdateAvailable() const { return m_hasUpdate; }
|
bool hasUpdateAvailable() const
|
||||||
|
{
|
||||||
|
return m_hasUpdate;
|
||||||
|
}
|
||||||
void setUpdateAvailable(bool value)
|
void setUpdateAvailable(bool value)
|
||||||
{
|
{
|
||||||
if (m_hasUpdate != value) {
|
if(m_hasUpdate != value)
|
||||||
|
{
|
||||||
m_hasUpdate = value;
|
m_hasUpdate = value;
|
||||||
emit propertiesChanged(this);
|
emit propertiesChanged(this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool hasCrashed() const { return m_crashed; }
|
bool hasCrashed() const
|
||||||
|
{
|
||||||
|
return m_crashed;
|
||||||
|
}
|
||||||
void setCrashed(bool value)
|
void setCrashed(bool value)
|
||||||
{
|
{
|
||||||
if (m_crashed != value) {
|
if(m_crashed != value)
|
||||||
|
{
|
||||||
m_crashed = value;
|
m_crashed = value;
|
||||||
emit propertiesChanged(this);
|
emit propertiesChanged(this);
|
||||||
}
|
}
|
||||||
@ -248,8 +270,6 @@ class BaseInstance : public QObject, public std::enable_shared_from_this<BaseIns
|
|||||||
virtual bool canEdit() const = 0;
|
virtual bool canEdit() const = 0;
|
||||||
virtual bool canExport() const = 0;
|
virtual bool canExport() const = 0;
|
||||||
|
|
||||||
virtual void populateLaunchMenu(QMenu* menu) = 0;
|
|
||||||
|
|
||||||
bool reloadSettings();
|
bool reloadSettings();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -268,32 +288,30 @@ class BaseInstance : public QObject, public std::enable_shared_from_this<BaseIns
|
|||||||
bool removeLinkedInstanceId(const QString& id);
|
bool removeLinkedInstanceId(const QString& id);
|
||||||
bool isLinkedToInstanceId(const QString& id) const;
|
bool isLinkedToInstanceId(const QString& id) const;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void changeStatus(Status newStatus);
|
void changeStatus(Status newStatus);
|
||||||
|
|
||||||
SettingsObjectPtr globalSettings() const { return m_global_settings.lock(); }
|
SettingsObjectPtr globalSettings() const { return m_global_settings.lock(); };
|
||||||
|
|
||||||
bool isSpecificSettingsLoaded() const { return m_specific_settings_loaded; }
|
bool isSpecificSettingsLoaded() const { return m_specific_settings_loaded; }
|
||||||
void setSpecificSettingsLoaded(bool loaded) { m_specific_settings_loaded = loaded; }
|
void setSpecificSettingsLoaded(bool loaded) { m_specific_settings_loaded = loaded; }
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
/*!
|
/*!
|
||||||
* \brief Signal emitted when properties relevant to the instance view change
|
* \brief Signal emitted when properties relevant to the instance view change
|
||||||
*/
|
*/
|
||||||
void propertiesChanged(BaseInstance* inst);
|
void propertiesChanged(BaseInstance *inst);
|
||||||
|
|
||||||
void launchTaskChanged(shared_qobject_ptr<LaunchTask>);
|
void launchTaskChanged(shared_qobject_ptr<LaunchTask>);
|
||||||
|
|
||||||
void runningStatusChanged(bool running);
|
void runningStatusChanged(bool running);
|
||||||
|
|
||||||
void profilerChanged();
|
|
||||||
|
|
||||||
void statusChanged(Status from, Status to);
|
void statusChanged(Status from, Status to);
|
||||||
|
|
||||||
protected slots:
|
protected slots:
|
||||||
void iconUpdated(QString key);
|
void iconUpdated(QString key);
|
||||||
|
|
||||||
protected: /* data */
|
protected: /* data */
|
||||||
QString m_rootDir;
|
QString m_rootDir;
|
||||||
SettingsObjectPtr m_settings;
|
SettingsObjectPtr m_settings;
|
||||||
// InstanceFlags m_flags;
|
// InstanceFlags m_flags;
|
||||||
@ -302,7 +320,7 @@ class BaseInstance : public QObject, public std::enable_shared_from_this<BaseIns
|
|||||||
QDateTime m_timeStarted;
|
QDateTime m_timeStarted;
|
||||||
RuntimeContext m_runtimeContext;
|
RuntimeContext m_runtimeContext;
|
||||||
|
|
||||||
private: /* data */
|
private: /* data */
|
||||||
Status m_status = Status::Present;
|
Status m_status = Status::Present;
|
||||||
bool m_crashed = false;
|
bool m_crashed = false;
|
||||||
bool m_hasUpdate = false;
|
bool m_hasUpdate = false;
|
||||||
@ -310,8 +328,9 @@ class BaseInstance : public QObject, public std::enable_shared_from_this<BaseIns
|
|||||||
|
|
||||||
SettingsObjectWeakPtr m_global_settings;
|
SettingsObjectWeakPtr m_global_settings;
|
||||||
bool m_specific_settings_loaded = false;
|
bool m_specific_settings_loaded = false;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
Q_DECLARE_METATYPE(shared_qobject_ptr<BaseInstance>)
|
Q_DECLARE_METATYPE(shared_qobject_ptr<BaseInstance>)
|
||||||
// Q_DECLARE_METATYPE(BaseInstance::InstanceFlag)
|
//Q_DECLARE_METATYPE(BaseInstance::InstanceFlag)
|
||||||
// Q_DECLARE_OPERATORS_FOR_FLAGS(BaseInstance::InstanceFlags)
|
//Q_DECLARE_OPERATORS_FOR_FLAGS(BaseInstance::InstanceFlags)
|
||||||
|
@ -43,8 +43,9 @@ class BaseVersion {
|
|||||||
* the kind of version this is (Stable, Beta, Snapshot, whatever)
|
* the kind of version this is (Stable, Beta, Snapshot, whatever)
|
||||||
*/
|
*/
|
||||||
virtual QString typeString() const = 0;
|
virtual QString typeString() const = 0;
|
||||||
virtual bool operator<(BaseVersion& a) { return name() < a.name(); }
|
|
||||||
virtual bool operator>(BaseVersion& a) { return name() > a.name(); }
|
virtual bool operator<(BaseVersion& a) { return name() < a.name(); };
|
||||||
|
virtual bool operator>(BaseVersion& a) { return name() > a.name(); };
|
||||||
};
|
};
|
||||||
|
|
||||||
Q_DECLARE_METATYPE(BaseVersion::Ptr)
|
Q_DECLARE_METATYPE(BaseVersion::Ptr)
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
@ -36,11 +36,14 @@
|
|||||||
#include "BaseVersionList.h"
|
#include "BaseVersionList.h"
|
||||||
#include "BaseVersion.h"
|
#include "BaseVersion.h"
|
||||||
|
|
||||||
BaseVersionList::BaseVersionList(QObject* parent) : QAbstractListModel(parent) {}
|
BaseVersionList::BaseVersionList(QObject *parent) : QAbstractListModel(parent)
|
||||||
|
|
||||||
BaseVersion::Ptr BaseVersionList::findVersion(const QString& descriptor)
|
|
||||||
{
|
{
|
||||||
for (int i = 0; i < count(); i++) {
|
}
|
||||||
|
|
||||||
|
BaseVersion::Ptr BaseVersionList::findVersion(const QString &descriptor)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < count(); i++)
|
||||||
|
{
|
||||||
if (at(i)->descriptor() == descriptor)
|
if (at(i)->descriptor() == descriptor)
|
||||||
return at(i);
|
return at(i);
|
||||||
}
|
}
|
||||||
@ -55,7 +58,7 @@ BaseVersion::Ptr BaseVersionList::getRecommended() const
|
|||||||
return at(0);
|
return at(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
QVariant BaseVersionList::data(const QModelIndex& index, int role) const
|
QVariant BaseVersionList::data(const QModelIndex &index, int role) const
|
||||||
{
|
{
|
||||||
if (!index.isValid())
|
if (!index.isValid())
|
||||||
return QVariant();
|
return QVariant();
|
||||||
@ -65,36 +68,37 @@ QVariant BaseVersionList::data(const QModelIndex& index, int role) const
|
|||||||
|
|
||||||
BaseVersion::Ptr version = at(index.row());
|
BaseVersion::Ptr version = at(index.row());
|
||||||
|
|
||||||
switch (role) {
|
switch (role)
|
||||||
case VersionPointerRole:
|
{
|
||||||
return QVariant::fromValue(version);
|
case VersionPointerRole:
|
||||||
|
return QVariant::fromValue(version);
|
||||||
|
|
||||||
case VersionRole:
|
case VersionRole:
|
||||||
return version->name();
|
return version->name();
|
||||||
|
|
||||||
case VersionIdRole:
|
case VersionIdRole:
|
||||||
return version->descriptor();
|
return version->descriptor();
|
||||||
|
|
||||||
case TypeRole:
|
case TypeRole:
|
||||||
return version->typeString();
|
return version->typeString();
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return QVariant();
|
return QVariant();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
BaseVersionList::RoleList BaseVersionList::providesRoles() const
|
BaseVersionList::RoleList BaseVersionList::providesRoles() const
|
||||||
{
|
{
|
||||||
return { VersionPointerRole, VersionRole, VersionIdRole, TypeRole };
|
return {VersionPointerRole, VersionRole, VersionIdRole, TypeRole};
|
||||||
}
|
}
|
||||||
|
|
||||||
int BaseVersionList::rowCount(const QModelIndex& parent) const
|
int BaseVersionList::rowCount(const QModelIndex &parent) const
|
||||||
{
|
{
|
||||||
// Return count
|
// Return count
|
||||||
return parent.isValid() ? 0 : count();
|
return parent.isValid() ? 0 : count();
|
||||||
}
|
}
|
||||||
|
|
||||||
int BaseVersionList::columnCount(const QModelIndex& parent) const
|
int BaseVersionList::columnCount(const QModelIndex &parent) const
|
||||||
{
|
{
|
||||||
return parent.isValid() ? 0 : 1;
|
return parent.isValid() ? 0 : 1;
|
||||||
}
|
}
|
||||||
|
@ -15,13 +15,13 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QAbstractListModel>
|
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QVariant>
|
#include <QVariant>
|
||||||
|
#include <QAbstractListModel>
|
||||||
|
|
||||||
#include "BaseVersion.h"
|
#include "BaseVersion.h"
|
||||||
#include "QObjectPtr.h"
|
|
||||||
#include "tasks/Task.h"
|
#include "tasks/Task.h"
|
||||||
|
#include "QObjectPtr.h"
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \brief Class that each instance type's version list derives from.
|
* \brief Class that each instance type's version list derives from.
|
||||||
@ -35,10 +35,12 @@
|
|||||||
* all have a default implementation, but they can be overridden by plugins to
|
* all have a default implementation, but they can be overridden by plugins to
|
||||||
* change the behavior of the list.
|
* change the behavior of the list.
|
||||||
*/
|
*/
|
||||||
class BaseVersionList : public QAbstractListModel {
|
class BaseVersionList : public QAbstractListModel
|
||||||
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
enum ModelRoles {
|
enum ModelRoles
|
||||||
|
{
|
||||||
VersionPointerRole = Qt::UserRole,
|
VersionPointerRole = Qt::UserRole,
|
||||||
VersionRole,
|
VersionRole,
|
||||||
VersionIdRole,
|
VersionIdRole,
|
||||||
@ -51,9 +53,9 @@ class BaseVersionList : public QAbstractListModel {
|
|||||||
ArchitectureRole,
|
ArchitectureRole,
|
||||||
SortRole
|
SortRole
|
||||||
};
|
};
|
||||||
using RoleList = QList<int>;
|
typedef QList<int> RoleList;
|
||||||
|
|
||||||
explicit BaseVersionList(QObject* parent = 0);
|
explicit BaseVersionList(QObject *parent = 0);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \brief Gets a task that will reload the version list.
|
* \brief Gets a task that will reload the version list.
|
||||||
@ -64,7 +66,7 @@ class BaseVersionList : public QAbstractListModel {
|
|||||||
virtual Task::Ptr getLoadTask() = 0;
|
virtual Task::Ptr getLoadTask() = 0;
|
||||||
|
|
||||||
//! Checks whether or not the list is loaded. If this returns false, the list should be
|
//! Checks whether or not the list is loaded. If this returns false, the list should be
|
||||||
// loaded.
|
//loaded.
|
||||||
virtual bool isLoaded() = 0;
|
virtual bool isLoaded() = 0;
|
||||||
|
|
||||||
//! Gets the version at the given index.
|
//! Gets the version at the given index.
|
||||||
@ -74,9 +76,9 @@ class BaseVersionList : public QAbstractListModel {
|
|||||||
virtual int count() const = 0;
|
virtual int count() const = 0;
|
||||||
|
|
||||||
//////// List Model Functions ////////
|
//////// List Model Functions ////////
|
||||||
QVariant data(const QModelIndex& index, int role) const override;
|
QVariant data(const QModelIndex &index, int role) const override;
|
||||||
int rowCount(const QModelIndex& parent) const override;
|
int rowCount(const QModelIndex &parent) const override;
|
||||||
int columnCount(const QModelIndex& parent) const override;
|
int columnCount(const QModelIndex &parent) const override;
|
||||||
QHash<int, QByteArray> roleNames() const override;
|
QHash<int, QByteArray> roleNames() const override;
|
||||||
|
|
||||||
//! which roles are provided by this version list?
|
//! which roles are provided by this version list?
|
||||||
@ -88,7 +90,7 @@ class BaseVersionList : public QAbstractListModel {
|
|||||||
* \return A const pointer to the version with the given descriptor. NULL if
|
* \return A const pointer to the version with the given descriptor. NULL if
|
||||||
* one doesn't exist.
|
* one doesn't exist.
|
||||||
*/
|
*/
|
||||||
virtual BaseVersion::Ptr findVersion(const QString& descriptor);
|
virtual BaseVersion::Ptr findVersion(const QString &descriptor);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* \brief Gets the recommended version from this list
|
* \brief Gets the recommended version from this list
|
||||||
@ -101,7 +103,8 @@ class BaseVersionList : public QAbstractListModel {
|
|||||||
*/
|
*/
|
||||||
virtual void sortVersions() = 0;
|
virtual void sortVersions() = 0;
|
||||||
|
|
||||||
protected slots:
|
protected
|
||||||
|
slots:
|
||||||
/*!
|
/*!
|
||||||
* Updates this list with the given list of versions.
|
* Updates this list with the given list of versions.
|
||||||
* This is done by copying each version in the given list and inserting it
|
* This is done by copying each version in the given list and inserting it
|
||||||
|
@ -136,16 +136,6 @@ set(NET_SOURCES
|
|||||||
net/Validator.h
|
net/Validator.h
|
||||||
net/Upload.cpp
|
net/Upload.cpp
|
||||||
net/Upload.h
|
net/Upload.h
|
||||||
net/HeaderProxy.h
|
|
||||||
net/RawHeaderProxy.h
|
|
||||||
net/ApiHeaderProxy.h
|
|
||||||
net/StaticHeaderProxy.h
|
|
||||||
net/ApiDownload.h
|
|
||||||
net/ApiDownload.cpp
|
|
||||||
net/ApiUpload.cpp
|
|
||||||
net/ApiUpload.h
|
|
||||||
net/NetRequest.cpp
|
|
||||||
net/NetRequest.h
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# Game launch logic
|
# Game launch logic
|
||||||
@ -182,11 +172,6 @@ set(MAC_UPDATE_SOURCES
|
|||||||
updater/MacSparkleUpdater.mm
|
updater/MacSparkleUpdater.mm
|
||||||
)
|
)
|
||||||
|
|
||||||
set(PRISM_UPDATE_SOURCES
|
|
||||||
updater/PrismExternalUpdater.h
|
|
||||||
updater/PrismExternalUpdater.cpp
|
|
||||||
)
|
|
||||||
|
|
||||||
# Backend for the news bar... there's usually no news.
|
# Backend for the news bar... there's usually no news.
|
||||||
set(NEWS_SOURCES
|
set(NEWS_SOURCES
|
||||||
# News System
|
# News System
|
||||||
@ -222,9 +207,13 @@ set(MINECRAFT_SOURCES
|
|||||||
minecraft/auth/MinecraftAccount.h
|
minecraft/auth/MinecraftAccount.h
|
||||||
minecraft/auth/Parsers.cpp
|
minecraft/auth/Parsers.cpp
|
||||||
minecraft/auth/Parsers.h
|
minecraft/auth/Parsers.h
|
||||||
|
minecraft/auth/Yggdrasil.cpp
|
||||||
|
minecraft/auth/Yggdrasil.h
|
||||||
|
|
||||||
minecraft/auth/flows/AuthFlow.cpp
|
minecraft/auth/flows/AuthFlow.cpp
|
||||||
minecraft/auth/flows/AuthFlow.h
|
minecraft/auth/flows/AuthFlow.h
|
||||||
|
minecraft/auth/flows/Mojang.cpp
|
||||||
|
minecraft/auth/flows/Mojang.h
|
||||||
minecraft/auth/flows/MSA.cpp
|
minecraft/auth/flows/MSA.cpp
|
||||||
minecraft/auth/flows/MSA.h
|
minecraft/auth/flows/MSA.h
|
||||||
minecraft/auth/flows/Offline.cpp
|
minecraft/auth/flows/Offline.cpp
|
||||||
@ -238,8 +227,12 @@ set(MINECRAFT_SOURCES
|
|||||||
minecraft/auth/steps/GetSkinStep.h
|
minecraft/auth/steps/GetSkinStep.h
|
||||||
minecraft/auth/steps/LauncherLoginStep.cpp
|
minecraft/auth/steps/LauncherLoginStep.cpp
|
||||||
minecraft/auth/steps/LauncherLoginStep.h
|
minecraft/auth/steps/LauncherLoginStep.h
|
||||||
|
minecraft/auth/steps/MigrationEligibilityStep.cpp
|
||||||
|
minecraft/auth/steps/MigrationEligibilityStep.h
|
||||||
minecraft/auth/steps/MinecraftProfileStep.cpp
|
minecraft/auth/steps/MinecraftProfileStep.cpp
|
||||||
minecraft/auth/steps/MinecraftProfileStep.h
|
minecraft/auth/steps/MinecraftProfileStep.h
|
||||||
|
minecraft/auth/steps/MinecraftProfileStepMojang.cpp
|
||||||
|
minecraft/auth/steps/MinecraftProfileStepMojang.h
|
||||||
minecraft/auth/steps/MSAStep.cpp
|
minecraft/auth/steps/MSAStep.cpp
|
||||||
minecraft/auth/steps/MSAStep.h
|
minecraft/auth/steps/MSAStep.h
|
||||||
minecraft/auth/steps/XboxAuthorizationStep.cpp
|
minecraft/auth/steps/XboxAuthorizationStep.cpp
|
||||||
@ -248,6 +241,8 @@ set(MINECRAFT_SOURCES
|
|||||||
minecraft/auth/steps/XboxProfileStep.h
|
minecraft/auth/steps/XboxProfileStep.h
|
||||||
minecraft/auth/steps/XboxUserStep.cpp
|
minecraft/auth/steps/XboxUserStep.cpp
|
||||||
minecraft/auth/steps/XboxUserStep.h
|
minecraft/auth/steps/XboxUserStep.h
|
||||||
|
minecraft/auth/steps/YggdrasilStep.cpp
|
||||||
|
minecraft/auth/steps/YggdrasilStep.h
|
||||||
|
|
||||||
minecraft/gameoptions/GameOptions.h
|
minecraft/gameoptions/GameOptions.h
|
||||||
minecraft/gameoptions/GameOptions.cpp
|
minecraft/gameoptions/GameOptions.cpp
|
||||||
@ -267,6 +262,8 @@ set(MINECRAFT_SOURCES
|
|||||||
minecraft/launch/CreateGameFolders.h
|
minecraft/launch/CreateGameFolders.h
|
||||||
minecraft/launch/ModMinecraftJar.cpp
|
minecraft/launch/ModMinecraftJar.cpp
|
||||||
minecraft/launch/ModMinecraftJar.h
|
minecraft/launch/ModMinecraftJar.h
|
||||||
|
minecraft/launch/DirectJavaLaunch.cpp
|
||||||
|
minecraft/launch/DirectJavaLaunch.h
|
||||||
minecraft/launch/ExtractNatives.cpp
|
minecraft/launch/ExtractNatives.cpp
|
||||||
minecraft/launch/ExtractNatives.h
|
minecraft/launch/ExtractNatives.h
|
||||||
minecraft/launch/LauncherPartLaunch.cpp
|
minecraft/launch/LauncherPartLaunch.cpp
|
||||||
@ -365,8 +362,6 @@ set(MINECRAFT_SOURCES
|
|||||||
minecraft/mod/tasks/LocalWorldSaveParseTask.cpp
|
minecraft/mod/tasks/LocalWorldSaveParseTask.cpp
|
||||||
minecraft/mod/tasks/LocalResourceParse.h
|
minecraft/mod/tasks/LocalResourceParse.h
|
||||||
minecraft/mod/tasks/LocalResourceParse.cpp
|
minecraft/mod/tasks/LocalResourceParse.cpp
|
||||||
minecraft/mod/tasks/GetModDependenciesTask.h
|
|
||||||
minecraft/mod/tasks/GetModDependenciesTask.cpp
|
|
||||||
|
|
||||||
# Assets
|
# Assets
|
||||||
minecraft/AssetsUtils.h
|
minecraft/AssetsUtils.h
|
||||||
@ -490,9 +485,6 @@ set(API_SOURCES
|
|||||||
modplatform/helpers/HashUtils.cpp
|
modplatform/helpers/HashUtils.cpp
|
||||||
modplatform/helpers/OverrideUtils.h
|
modplatform/helpers/OverrideUtils.h
|
||||||
modplatform/helpers/OverrideUtils.cpp
|
modplatform/helpers/OverrideUtils.cpp
|
||||||
|
|
||||||
modplatform/helpers/ExportToModList.h
|
|
||||||
modplatform/helpers/ExportToModList.cpp
|
|
||||||
)
|
)
|
||||||
|
|
||||||
set(FTB_SOURCES
|
set(FTB_SOURCES
|
||||||
@ -504,11 +496,6 @@ set(FTB_SOURCES
|
|||||||
modplatform/legacy_ftb/PrivatePackManager.cpp
|
modplatform/legacy_ftb/PrivatePackManager.cpp
|
||||||
|
|
||||||
modplatform/legacy_ftb/PackHelpers.h
|
modplatform/legacy_ftb/PackHelpers.h
|
||||||
|
|
||||||
modplatform/import_ftb/PackInstallTask.h
|
|
||||||
modplatform/import_ftb/PackInstallTask.cpp
|
|
||||||
modplatform/import_ftb/PackHelpers.h
|
|
||||||
modplatform/import_ftb/PackHelpers.cpp
|
|
||||||
)
|
)
|
||||||
|
|
||||||
set(FLAME_SOURCES
|
set(FLAME_SOURCES
|
||||||
@ -525,8 +512,6 @@ set(FLAME_SOURCES
|
|||||||
modplatform/flame/FlameCheckUpdate.h
|
modplatform/flame/FlameCheckUpdate.h
|
||||||
modplatform/flame/FlameInstanceCreationTask.h
|
modplatform/flame/FlameInstanceCreationTask.h
|
||||||
modplatform/flame/FlameInstanceCreationTask.cpp
|
modplatform/flame/FlameInstanceCreationTask.cpp
|
||||||
modplatform/flame/FlamePackExportTask.h
|
|
||||||
modplatform/flame/FlamePackExportTask.cpp
|
|
||||||
)
|
)
|
||||||
|
|
||||||
set(MODRINTH_SOURCES
|
set(MODRINTH_SOURCES
|
||||||
@ -571,9 +556,6 @@ set(ATLAUNCHER_SOURCES
|
|||||||
)
|
)
|
||||||
|
|
||||||
set(LINKEXE_SOURCES
|
set(LINKEXE_SOURCES
|
||||||
WindowsConsole.cpp
|
|
||||||
WindowsConsole.h
|
|
||||||
|
|
||||||
filelink/FileLink.h
|
filelink/FileLink.h
|
||||||
filelink/FileLink.cpp
|
filelink/FileLink.cpp
|
||||||
FileSystem.h
|
FileSystem.h
|
||||||
@ -585,63 +567,6 @@ set(LINKEXE_SOURCES
|
|||||||
DesktopServices.cpp
|
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 ########
|
######## Logging categories ########
|
||||||
|
|
||||||
ecm_qt_declare_logging_category(CORE_SOURCES
|
ecm_qt_declare_logging_category(CORE_SOURCES
|
||||||
@ -736,10 +661,8 @@ set(LOGIC_SOURCES
|
|||||||
${ATLAUNCHER_SOURCES}
|
${ATLAUNCHER_SOURCES}
|
||||||
)
|
)
|
||||||
|
|
||||||
if(APPLE AND Launcher_ENABLE_UPDATER)
|
if(APPLE)
|
||||||
set (LOGIC_SOURCES ${LOGIC_SOURCES} ${MAC_UPDATE_SOURCES})
|
set (LOGIC_SOURCES ${LOGIC_SOURCES} ${MAC_UPDATE_SOURCES})
|
||||||
else()
|
|
||||||
set (LOGIC_SOURCES ${LOGIC_SOURCES} ${PRISM_UPDATE_SOURCES})
|
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
SET(LAUNCHER_SOURCES
|
SET(LAUNCHER_SOURCES
|
||||||
@ -829,12 +752,8 @@ SET(LAUNCHER_SOURCES
|
|||||||
ui/themes/ITheme.h
|
ui/themes/ITheme.h
|
||||||
ui/themes/SystemTheme.cpp
|
ui/themes/SystemTheme.cpp
|
||||||
ui/themes/SystemTheme.h
|
ui/themes/SystemTheme.h
|
||||||
ui/themes/IconTheme.cpp
|
|
||||||
ui/themes/IconTheme.h
|
|
||||||
ui/themes/ThemeManager.cpp
|
ui/themes/ThemeManager.cpp
|
||||||
ui/themes/ThemeManager.h
|
ui/themes/ThemeManager.h
|
||||||
ui/themes/CatPack.cpp
|
|
||||||
ui/themes/CatPack.h
|
|
||||||
|
|
||||||
# Processes
|
# Processes
|
||||||
LaunchController.h
|
LaunchController.h
|
||||||
@ -889,8 +808,6 @@ SET(LAUNCHER_SOURCES
|
|||||||
ui/pages/global/AccountListPage.h
|
ui/pages/global/AccountListPage.h
|
||||||
ui/pages/global/CustomCommandsPage.cpp
|
ui/pages/global/CustomCommandsPage.cpp
|
||||||
ui/pages/global/CustomCommandsPage.h
|
ui/pages/global/CustomCommandsPage.h
|
||||||
ui/pages/global/EnvironmentVariablesPage.cpp
|
|
||||||
ui/pages/global/EnvironmentVariablesPage.h
|
|
||||||
ui/pages/global/ExternalToolsPage.cpp
|
ui/pages/global/ExternalToolsPage.cpp
|
||||||
ui/pages/global/ExternalToolsPage.h
|
ui/pages/global/ExternalToolsPage.h
|
||||||
ui/pages/global/JavaPage.cpp
|
ui/pages/global/JavaPage.cpp
|
||||||
@ -946,11 +863,6 @@ SET(LAUNCHER_SOURCES
|
|||||||
ui/pages/modplatform/legacy_ftb/ListModel.h
|
ui/pages/modplatform/legacy_ftb/ListModel.h
|
||||||
ui/pages/modplatform/legacy_ftb/ListModel.cpp
|
ui/pages/modplatform/legacy_ftb/ListModel.cpp
|
||||||
|
|
||||||
ui/pages/modplatform/import_ftb/ImportFTBPage.cpp
|
|
||||||
ui/pages/modplatform/import_ftb/ImportFTBPage.h
|
|
||||||
ui/pages/modplatform/import_ftb/ListModel.h
|
|
||||||
ui/pages/modplatform/import_ftb/ListModel.cpp
|
|
||||||
|
|
||||||
ui/pages/modplatform/flame/FlameModel.cpp
|
ui/pages/modplatform/flame/FlameModel.cpp
|
||||||
ui/pages/modplatform/flame/FlameModel.h
|
ui/pages/modplatform/flame/FlameModel.h
|
||||||
ui/pages/modplatform/flame/FlamePage.cpp
|
ui/pages/modplatform/flame/FlamePage.cpp
|
||||||
@ -973,9 +885,6 @@ SET(LAUNCHER_SOURCES
|
|||||||
ui/pages/modplatform/ImportPage.cpp
|
ui/pages/modplatform/ImportPage.cpp
|
||||||
ui/pages/modplatform/ImportPage.h
|
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.cpp
|
||||||
ui/pages/modplatform/modrinth/ModrinthResourceModels.h
|
ui/pages/modplatform/modrinth/ModrinthResourceModels.h
|
||||||
ui/pages/modplatform/modrinth/ModrinthResourcePages.cpp
|
ui/pages/modplatform/modrinth/ModrinthResourcePages.cpp
|
||||||
@ -996,14 +905,14 @@ SET(LAUNCHER_SOURCES
|
|||||||
ui/dialogs/EditAccountDialog.h
|
ui/dialogs/EditAccountDialog.h
|
||||||
ui/dialogs/ExportInstanceDialog.cpp
|
ui/dialogs/ExportInstanceDialog.cpp
|
||||||
ui/dialogs/ExportInstanceDialog.h
|
ui/dialogs/ExportInstanceDialog.h
|
||||||
ui/dialogs/ExportPackDialog.cpp
|
ui/dialogs/ExportMrPackDialog.cpp
|
||||||
ui/dialogs/ExportPackDialog.h
|
ui/dialogs/ExportMrPackDialog.h
|
||||||
ui/dialogs/ExportToModListDialog.cpp
|
|
||||||
ui/dialogs/ExportToModListDialog.h
|
|
||||||
ui/dialogs/IconPickerDialog.cpp
|
ui/dialogs/IconPickerDialog.cpp
|
||||||
ui/dialogs/IconPickerDialog.h
|
ui/dialogs/IconPickerDialog.h
|
||||||
ui/dialogs/ImportResourceDialog.cpp
|
ui/dialogs/ImportResourceDialog.cpp
|
||||||
ui/dialogs/ImportResourceDialog.h
|
ui/dialogs/ImportResourceDialog.h
|
||||||
|
ui/dialogs/LoginDialog.cpp
|
||||||
|
ui/dialogs/LoginDialog.h
|
||||||
ui/dialogs/MSALoginDialog.cpp
|
ui/dialogs/MSALoginDialog.cpp
|
||||||
ui/dialogs/MSALoginDialog.h
|
ui/dialogs/MSALoginDialog.h
|
||||||
ui/dialogs/OfflineLoginDialog.cpp
|
ui/dialogs/OfflineLoginDialog.cpp
|
||||||
@ -1034,16 +943,12 @@ SET(LAUNCHER_SOURCES
|
|||||||
ui/dialogs/ChooseProviderDialog.cpp
|
ui/dialogs/ChooseProviderDialog.cpp
|
||||||
ui/dialogs/ModUpdateDialog.cpp
|
ui/dialogs/ModUpdateDialog.cpp
|
||||||
ui/dialogs/ModUpdateDialog.h
|
ui/dialogs/ModUpdateDialog.h
|
||||||
ui/dialogs/InstallLoaderDialog.cpp
|
|
||||||
ui/dialogs/InstallLoaderDialog.h
|
|
||||||
|
|
||||||
# GUI - widgets
|
# GUI - widgets
|
||||||
ui/widgets/Common.cpp
|
ui/widgets/Common.cpp
|
||||||
ui/widgets/Common.h
|
ui/widgets/Common.h
|
||||||
ui/widgets/CustomCommands.cpp
|
ui/widgets/CustomCommands.cpp
|
||||||
ui/widgets/CustomCommands.h
|
ui/widgets/CustomCommands.h
|
||||||
ui/widgets/EnvironmentVariables.cpp
|
|
||||||
ui/widgets/EnvironmentVariables.h
|
|
||||||
ui/widgets/DropLabel.cpp
|
ui/widgets/DropLabel.cpp
|
||||||
ui/widgets/DropLabel.h
|
ui/widgets/DropLabel.h
|
||||||
ui/widgets/FocusLineEdit.cpp
|
ui/widgets/FocusLineEdit.cpp
|
||||||
@ -1102,23 +1007,6 @@ SET(LAUNCHER_SOURCES
|
|||||||
ui/instanceview/VisualGroup.h
|
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
|
|
||||||
WindowsConsole.h
|
|
||||||
${LAUNCHER_SOURCES}
|
|
||||||
)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
qt_wrap_ui(LAUNCHER_UI
|
qt_wrap_ui(LAUNCHER_UI
|
||||||
ui/MainWindow.ui
|
ui/MainWindow.ui
|
||||||
ui/setupwizard/PasteWizardPage.ui
|
ui/setupwizard/PasteWizardPage.ui
|
||||||
@ -1147,14 +1035,11 @@ qt_wrap_ui(LAUNCHER_UI
|
|||||||
ui/pages/modplatform/ResourcePage.ui
|
ui/pages/modplatform/ResourcePage.ui
|
||||||
ui/pages/modplatform/flame/FlamePage.ui
|
ui/pages/modplatform/flame/FlamePage.ui
|
||||||
ui/pages/modplatform/legacy_ftb/Page.ui
|
ui/pages/modplatform/legacy_ftb/Page.ui
|
||||||
ui/pages/modplatform/import_ftb/ImportFTBPage.ui
|
|
||||||
ui/pages/modplatform/ImportPage.ui
|
ui/pages/modplatform/ImportPage.ui
|
||||||
ui/pages/modplatform/OptionalModDialog.ui
|
|
||||||
ui/pages/modplatform/modrinth/ModrinthPage.ui
|
ui/pages/modplatform/modrinth/ModrinthPage.ui
|
||||||
ui/pages/modplatform/technic/TechnicPage.ui
|
ui/pages/modplatform/technic/TechnicPage.ui
|
||||||
ui/widgets/InstanceCardWidget.ui
|
ui/widgets/InstanceCardWidget.ui
|
||||||
ui/widgets/CustomCommands.ui
|
ui/widgets/CustomCommands.ui
|
||||||
ui/widgets/EnvironmentVariables.ui
|
|
||||||
ui/widgets/InfoFrame.ui
|
ui/widgets/InfoFrame.ui
|
||||||
ui/widgets/ModFilterWidget.ui
|
ui/widgets/ModFilterWidget.ui
|
||||||
ui/widgets/SubTaskProgressBar.ui
|
ui/widgets/SubTaskProgressBar.ui
|
||||||
@ -1168,13 +1053,13 @@ qt_wrap_ui(LAUNCHER_UI
|
|||||||
ui/dialogs/ProfileSelectDialog.ui
|
ui/dialogs/ProfileSelectDialog.ui
|
||||||
ui/dialogs/SkinUploadDialog.ui
|
ui/dialogs/SkinUploadDialog.ui
|
||||||
ui/dialogs/ExportInstanceDialog.ui
|
ui/dialogs/ExportInstanceDialog.ui
|
||||||
ui/dialogs/ExportPackDialog.ui
|
ui/dialogs/ExportMrPackDialog.ui
|
||||||
ui/dialogs/ExportToModListDialog.ui
|
|
||||||
ui/dialogs/IconPickerDialog.ui
|
ui/dialogs/IconPickerDialog.ui
|
||||||
ui/dialogs/ImportResourceDialog.ui
|
ui/dialogs/ImportResourceDialog.ui
|
||||||
ui/dialogs/MSALoginDialog.ui
|
ui/dialogs/MSALoginDialog.ui
|
||||||
ui/dialogs/OfflineLoginDialog.ui
|
ui/dialogs/OfflineLoginDialog.ui
|
||||||
ui/dialogs/AboutDialog.ui
|
ui/dialogs/AboutDialog.ui
|
||||||
|
ui/dialogs/LoginDialog.ui
|
||||||
ui/dialogs/EditAccountDialog.ui
|
ui/dialogs/EditAccountDialog.ui
|
||||||
ui/dialogs/ReviewMessageBox.ui
|
ui/dialogs/ReviewMessageBox.ui
|
||||||
ui/dialogs/ScrollMessageBox.ui
|
ui/dialogs/ScrollMessageBox.ui
|
||||||
@ -1182,14 +1067,6 @@ qt_wrap_ui(LAUNCHER_UI
|
|||||||
ui/dialogs/ChooseProviderDialog.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
|
qt_add_resources(LAUNCHER_RESOURCES
|
||||||
resources/backgrounds/backgrounds.qrc
|
resources/backgrounds/backgrounds.qrc
|
||||||
resources/multimc/multimc.qrc
|
resources/multimc/multimc.qrc
|
||||||
@ -1206,31 +1083,14 @@ qt_add_resources(LAUNCHER_RESOURCES
|
|||||||
../${Launcher_Branding_LogoQRC}
|
../${Launcher_Branding_LogoQRC}
|
||||||
)
|
)
|
||||||
|
|
||||||
qt_wrap_ui(PRISMUPDATER_UI
|
|
||||||
updater/prismupdater/SelectReleaseDialog.ui
|
|
||||||
ui/widgets/SubTaskProgressBar.ui
|
|
||||||
ui/dialogs/ProgressDialog.ui
|
|
||||||
)
|
|
||||||
|
|
||||||
######## Windows resource files ########
|
######## Windows resource files ########
|
||||||
if(WIN32)
|
if(WIN32)
|
||||||
set(LAUNCHER_RCS ${CMAKE_CURRENT_BINARY_DIR}/../${Launcher_Branding_WindowsRC})
|
set(LAUNCHER_RCS ${CMAKE_CURRENT_BINARY_DIR}/../${Launcher_Branding_WindowsRC})
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
include(CompilerWarnings)
|
|
||||||
|
|
||||||
# Add executable
|
# Add executable
|
||||||
add_library(Launcher_logic STATIC ${LOGIC_SOURCES} ${LAUNCHER_SOURCES} ${LAUNCHER_UI} ${LAUNCHER_RESOURCES})
|
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_include_directories(Launcher_logic PUBLIC ${CMAKE_CURRENT_SOURCE_DIR})
|
||||||
target_compile_definitions(Launcher_logic PUBLIC LAUNCHER_APPLICATION)
|
|
||||||
target_link_libraries(Launcher_logic
|
target_link_libraries(Launcher_logic
|
||||||
systeminfo
|
systeminfo
|
||||||
Launcher_murmur2
|
Launcher_murmur2
|
||||||
@ -1269,23 +1129,17 @@ if(APPLE)
|
|||||||
set(CMAKE_MACOSX_RPATH 1)
|
set(CMAKE_MACOSX_RPATH 1)
|
||||||
set(CMAKE_INSTALL_RPATH "@loader_path/../Frameworks/")
|
set(CMAKE_INSTALL_RPATH "@loader_path/../Frameworks/")
|
||||||
|
|
||||||
if(Launcher_ENABLE_UPDATER)
|
file(DOWNLOAD ${MACOSX_SPARKLE_DOWNLOAD_URL} ${CMAKE_BINARY_DIR}/Sparkle.tar.xz EXPECTED_HASH SHA256=${MACOSX_SPARKLE_SHA256})
|
||||||
file(DOWNLOAD ${MACOSX_SPARKLE_DOWNLOAD_URL} ${CMAKE_BINARY_DIR}/Sparkle.tar.xz EXPECTED_HASH SHA256=${MACOSX_SPARKLE_SHA256})
|
file(ARCHIVE_EXTRACT INPUT ${CMAKE_BINARY_DIR}/Sparkle.tar.xz DESTINATION ${CMAKE_BINARY_DIR}/frameworks/Sparkle)
|
||||||
file(ARCHIVE_EXTRACT INPUT ${CMAKE_BINARY_DIR}/Sparkle.tar.xz DESTINATION ${CMAKE_BINARY_DIR}/frameworks/Sparkle)
|
|
||||||
|
|
||||||
find_library(SPARKLE_FRAMEWORK Sparkle "${CMAKE_BINARY_DIR}/frameworks/Sparkle")
|
|
||||||
add_compile_definitions(SPARKLE_ENABLED)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
|
find_library(SPARKLE_FRAMEWORK Sparkle "${CMAKE_BINARY_DIR}/frameworks/Sparkle")
|
||||||
target_link_libraries(Launcher_logic
|
target_link_libraries(Launcher_logic
|
||||||
"-framework AppKit"
|
"-framework AppKit"
|
||||||
"-framework Carbon"
|
"-framework Carbon"
|
||||||
"-framework Foundation"
|
"-framework Foundation"
|
||||||
"-framework ApplicationServices"
|
"-framework ApplicationServices"
|
||||||
)
|
)
|
||||||
if(Launcher_ENABLE_UPDATER)
|
target_link_libraries(Launcher_logic ${SPARKLE_FRAMEWORK})
|
||||||
target_link_libraries(Launcher_logic ${SPARKLE_FRAMEWORK})
|
|
||||||
endif()
|
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
target_link_libraries(Launcher_logic)
|
target_link_libraries(Launcher_logic)
|
||||||
@ -1312,51 +1166,8 @@ install(TARGETS ${Launcher_Name}
|
|||||||
FRAMEWORK DESTINATION ${FRAMEWORK_DEST_DIR} COMPONENT Runtime
|
FRAMEWORK DESTINATION ${FRAMEWORK_DEST_DIR} COMPONENT Runtime
|
||||||
)
|
)
|
||||||
|
|
||||||
if(Launcher_BUILD_UPDATER)
|
if(WIN32)
|
||||||
# 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})
|
add_library(filelink_logic STATIC ${LINKEXE_SOURCES})
|
||||||
set_project_warnings(filelink_logic
|
|
||||||
"${Launcher_MSVC_WARNINGS}"
|
|
||||||
"${Launcher_CLANG_WARNINGS}"
|
|
||||||
"${Launcher_GCC_WARNINGS}")
|
|
||||||
|
|
||||||
target_include_directories(filelink_logic PUBLIC ${CMAKE_CURRENT_SOURCE_DIR})
|
target_include_directories(filelink_logic PUBLIC ${CMAKE_CURRENT_SOURCE_DIR})
|
||||||
target_link_libraries(filelink_logic
|
target_link_libraries(filelink_logic
|
||||||
systeminfo
|
systeminfo
|
||||||
@ -1369,7 +1180,7 @@ if(WIN32 OR (DEFINED Launcher_BUILD_FILELINKER AND Launcher_BUILD_FILELINKER))
|
|||||||
${Launcher_QT_LIBS}
|
${Launcher_QT_LIBS}
|
||||||
)
|
)
|
||||||
|
|
||||||
add_executable("${Launcher_Name}_filelink" WIN32 filelink/filelink_main.cpp)
|
add_executable("${Launcher_Name}_filelink" WIN32 filelink/main.cpp)
|
||||||
|
|
||||||
target_sources("${Launcher_Name}_filelink" PRIVATE filelink/filelink.exe.manifest)
|
target_sources("${Launcher_Name}_filelink" PRIVATE filelink/filelink.exe.manifest)
|
||||||
|
|
||||||
@ -1390,7 +1201,7 @@ if(WIN32 OR (DEFINED Launcher_BUILD_FILELINKER AND Launcher_BUILD_FILELINKER))
|
|||||||
)
|
)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if (UNIX AND APPLE AND Launcher_ENABLE_UPDATER)
|
if (UNIX AND APPLE)
|
||||||
# Add Sparkle updater
|
# Add Sparkle updater
|
||||||
# It has to be copied here instead of just allowing fixup_bundle to install it, otherwise essential parts of
|
# It has to be copied here instead of just allowing fixup_bundle to install it, otherwise essential parts of
|
||||||
# the framework aren't installed
|
# the framework aren't installed
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
@ -41,7 +41,8 @@
|
|||||||
* @file libutil/src/cmdutils.cpp
|
* @file libutil/src/cmdutils.cpp
|
||||||
*/
|
*/
|
||||||
|
|
||||||
namespace Commandline {
|
namespace Commandline
|
||||||
|
{
|
||||||
|
|
||||||
// commandline splitter
|
// commandline splitter
|
||||||
QStringList splitArgs(QString args)
|
QStringList splitArgs(QString args)
|
||||||
@ -50,15 +51,19 @@ QStringList splitArgs(QString args)
|
|||||||
QString current;
|
QString current;
|
||||||
bool escape = false;
|
bool escape = false;
|
||||||
QChar inquotes;
|
QChar inquotes;
|
||||||
for (int i = 0; i < args.length(); i++) {
|
for (int i = 0; i < args.length(); i++)
|
||||||
|
{
|
||||||
QChar cchar = args.at(i);
|
QChar cchar = args.at(i);
|
||||||
|
|
||||||
// \ escaped
|
// \ escaped
|
||||||
if (escape) {
|
if (escape)
|
||||||
|
{
|
||||||
current += cchar;
|
current += cchar;
|
||||||
escape = false;
|
escape = false;
|
||||||
// in "quotes"
|
// in "quotes"
|
||||||
} else if (!inquotes.isNull()) {
|
}
|
||||||
|
else if (!inquotes.isNull())
|
||||||
|
{
|
||||||
if (cchar == '\\')
|
if (cchar == '\\')
|
||||||
escape = true;
|
escape = true;
|
||||||
else if (cchar == inquotes)
|
else if (cchar == inquotes)
|
||||||
@ -66,13 +71,18 @@ QStringList splitArgs(QString args)
|
|||||||
else
|
else
|
||||||
current += cchar;
|
current += cchar;
|
||||||
// otherwise
|
// otherwise
|
||||||
} else {
|
}
|
||||||
if (cchar == ' ') {
|
else
|
||||||
if (!current.isEmpty()) {
|
{
|
||||||
|
if (cchar == ' ')
|
||||||
|
{
|
||||||
|
if (!current.isEmpty())
|
||||||
|
{
|
||||||
argv << current;
|
argv << current;
|
||||||
current.clear();
|
current.clear();
|
||||||
}
|
}
|
||||||
} else if (cchar == '"' || cchar == '\'')
|
}
|
||||||
|
else if (cchar == '"' || cchar == '\'')
|
||||||
inquotes = cchar;
|
inquotes = cchar;
|
||||||
else
|
else
|
||||||
current += cchar;
|
current += cchar;
|
||||||
@ -82,4 +92,4 @@ QStringList splitArgs(QString args)
|
|||||||
argv << current;
|
argv << current;
|
||||||
return argv;
|
return argv;
|
||||||
}
|
}
|
||||||
} // namespace Commandline
|
}
|
||||||
|
@ -25,7 +25,8 @@
|
|||||||
* @brief commandline parsing and processing utilities
|
* @brief commandline parsing and processing utilities
|
||||||
*/
|
*/
|
||||||
|
|
||||||
namespace Commandline {
|
namespace Commandline
|
||||||
|
{
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief split a string into argv items like a shell would do
|
* @brief split a string into argv items like a shell would do
|
||||||
@ -33,4 +34,4 @@ namespace Commandline {
|
|||||||
* @return a QStringList containing all arguments
|
* @return a QStringList containing all arguments
|
||||||
*/
|
*/
|
||||||
QStringList splitArgs(QString args);
|
QStringList splitArgs(QString args);
|
||||||
} // namespace Commandline
|
}
|
||||||
|
@ -1,21 +1,33 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
class DefaultVariable {
|
class DefaultVariable
|
||||||
public:
|
{
|
||||||
DefaultVariable(const T& value) { defaultValue = value; }
|
public:
|
||||||
DefaultVariable<T>& operator=(const T& value)
|
DefaultVariable(const T & value)
|
||||||
|
{
|
||||||
|
defaultValue = value;
|
||||||
|
}
|
||||||
|
DefaultVariable<T> & operator =(const T & value)
|
||||||
{
|
{
|
||||||
currentValue = value;
|
currentValue = value;
|
||||||
is_default = currentValue == defaultValue;
|
is_default = currentValue == defaultValue;
|
||||||
is_explicit = true;
|
is_explicit = true;
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
operator const T&() const { return is_default ? defaultValue : currentValue; }
|
operator const T &() const
|
||||||
bool isDefault() const { return is_default; }
|
{
|
||||||
bool isExplicit() const { return is_explicit; }
|
return is_default ? defaultValue : currentValue;
|
||||||
|
}
|
||||||
private:
|
bool isDefault() const
|
||||||
|
{
|
||||||
|
return is_default;
|
||||||
|
}
|
||||||
|
bool isExplicit() const
|
||||||
|
{
|
||||||
|
return is_explicit;
|
||||||
|
}
|
||||||
|
private:
|
||||||
T currentValue;
|
T currentValue;
|
||||||
T defaultValue;
|
T defaultValue;
|
||||||
bool is_default = true;
|
bool is_default = true;
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 dada513 <dada513@protonmail.com>
|
* Copyright (C) 2022 dada513 <dada513@protonmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
@ -33,37 +33,40 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
#include "DesktopServices.h"
|
#include "DesktopServices.h"
|
||||||
#include <QDebug>
|
|
||||||
#include <QDesktopServices>
|
|
||||||
#include <QDir>
|
#include <QDir>
|
||||||
|
#include <QDesktopServices>
|
||||||
#include <QProcess>
|
#include <QProcess>
|
||||||
|
#include <QDebug>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This shouldn't exist, but until QTBUG-9328 and other unreported bugs are fixed, it needs to be a thing.
|
* This shouldn't exist, but until QTBUG-9328 and other unreported bugs are fixed, it needs to be a thing.
|
||||||
*/
|
*/
|
||||||
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
||||||
|
|
||||||
|
#include <unistd.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
#include <sys/wait.h>
|
#include <sys/wait.h>
|
||||||
#include <unistd.h>
|
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
bool IndirectOpen(T callable, qint64* pid_forked = nullptr)
|
bool IndirectOpen(T callable, qint64 *pid_forked = nullptr)
|
||||||
{
|
{
|
||||||
auto pid = fork();
|
auto pid = fork();
|
||||||
if (pid_forked) {
|
if(pid_forked)
|
||||||
if (pid > 0)
|
{
|
||||||
|
if(pid > 0)
|
||||||
*pid_forked = pid;
|
*pid_forked = pid;
|
||||||
else
|
else
|
||||||
*pid_forked = 0;
|
*pid_forked = 0;
|
||||||
}
|
}
|
||||||
if (pid == -1) {
|
if(pid == -1)
|
||||||
|
{
|
||||||
qWarning() << "IndirectOpen failed to fork: " << errno;
|
qWarning() << "IndirectOpen failed to fork: " << errno;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
// child - do the stuff
|
// child - do the stuff
|
||||||
if (pid == 0) {
|
if(pid == 0)
|
||||||
|
{
|
||||||
// unset all this garbage so it doesn't get passed to the child process
|
// unset all this garbage so it doesn't get passed to the child process
|
||||||
qunsetenv("LD_PRELOAD");
|
qunsetenv("LD_PRELOAD");
|
||||||
qunsetenv("LD_LIBRARY_PATH");
|
qunsetenv("LD_LIBRARY_PATH");
|
||||||
@ -79,14 +82,19 @@ bool IndirectOpen(T callable, qint64* pid_forked = nullptr)
|
|||||||
|
|
||||||
// die. now. do not clean up anything, it would just hang forever.
|
// die. now. do not clean up anything, it would just hang forever.
|
||||||
_exit(status ? 0 : 1);
|
_exit(status ? 0 : 1);
|
||||||
} else {
|
}
|
||||||
// parent - assume it worked.
|
else
|
||||||
|
{
|
||||||
|
//parent - assume it worked.
|
||||||
int status;
|
int status;
|
||||||
while (waitpid(pid, &status, 0)) {
|
while (waitpid(pid, &status, 0))
|
||||||
if (WIFEXITED(status)) {
|
{
|
||||||
|
if(WIFEXITED(status))
|
||||||
|
{
|
||||||
return WEXITSTATUS(status) == 0;
|
return WEXITSTATUS(status) == 0;
|
||||||
}
|
}
|
||||||
if (WIFSIGNALED(status)) {
|
if(WIFSIGNALED(status))
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -96,19 +104,26 @@ bool IndirectOpen(T callable, qint64* pid_forked = nullptr)
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
namespace DesktopServices {
|
namespace DesktopServices {
|
||||||
bool openDirectory(const QString& path, [[maybe_unused]] bool ensureExists)
|
bool openDirectory(const QString &path, bool ensureExists)
|
||||||
{
|
{
|
||||||
qDebug() << "Opening directory" << path;
|
qDebug() << "Opening directory" << path;
|
||||||
QDir parentPath;
|
QDir parentPath;
|
||||||
QDir dir(path);
|
QDir dir(path);
|
||||||
if (ensureExists && !dir.exists()) {
|
if (!dir.exists())
|
||||||
|
{
|
||||||
parentPath.mkpath(dir.absolutePath());
|
parentPath.mkpath(dir.absolutePath());
|
||||||
}
|
}
|
||||||
auto f = [&]() { return QDesktopServices::openUrl(QUrl::fromLocalFile(dir.absolutePath())); };
|
auto f = [&]()
|
||||||
|
{
|
||||||
|
return QDesktopServices::openUrl(QUrl::fromLocalFile(dir.absolutePath()));
|
||||||
|
};
|
||||||
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
||||||
if (!isSandbox()) {
|
if(!isFlatpak())
|
||||||
|
{
|
||||||
return IndirectOpen(f);
|
return IndirectOpen(f);
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
return f();
|
return f();
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
@ -116,14 +131,20 @@ bool openDirectory(const QString& path, [[maybe_unused]] bool ensureExists)
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
bool openFile(const QString& path)
|
bool openFile(const QString &path)
|
||||||
{
|
{
|
||||||
qDebug() << "Opening file" << path;
|
qDebug() << "Opening file" << path;
|
||||||
auto f = [&]() { return QDesktopServices::openUrl(QUrl::fromLocalFile(path)); };
|
auto f = [&]()
|
||||||
|
{
|
||||||
|
return QDesktopServices::openUrl(QUrl::fromLocalFile(path));
|
||||||
|
};
|
||||||
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
||||||
if (!isSandbox()) {
|
if(!isFlatpak())
|
||||||
|
{
|
||||||
return IndirectOpen(f);
|
return IndirectOpen(f);
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
return f();
|
return f();
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
@ -131,29 +152,41 @@ bool openFile(const QString& path)
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
bool openFile(const QString& application, const QString& path, const QString& workingDirectory, qint64* pid)
|
bool openFile(const QString &application, const QString &path, const QString &workingDirectory, qint64 *pid)
|
||||||
{
|
{
|
||||||
qDebug() << "Opening file" << path << "using" << application;
|
qDebug() << "Opening file" << path << "using" << application;
|
||||||
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
||||||
// FIXME: the pid here is fake. So if something depends on it, it will likely misbehave
|
// FIXME: the pid here is fake. So if something depends on it, it will likely misbehave
|
||||||
if (!isSandbox()) {
|
if(!isFlatpak())
|
||||||
return IndirectOpen([&]() { return QProcess::startDetached(application, QStringList() << path, workingDirectory); }, pid);
|
{
|
||||||
} else {
|
return IndirectOpen([&]()
|
||||||
return QProcess::startDetached(application, QStringList() << path, workingDirectory, pid);
|
{
|
||||||
|
return QProcess::startDetached(application, QStringList() << path, workingDirectory);
|
||||||
|
}, pid);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return QProcess::startDetached(application, QStringList() << path, workingDirectory, pid);
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
return QProcess::startDetached(application, QStringList() << path, workingDirectory, pid);
|
return QProcess::startDetached(application, QStringList() << path, workingDirectory, pid);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
bool run(const QString& application, const QStringList& args, const QString& workingDirectory, qint64* pid)
|
bool run(const QString &application, const QStringList &args, const QString &workingDirectory, qint64 *pid)
|
||||||
{
|
{
|
||||||
qDebug() << "Running" << application << "with args" << args.join(' ');
|
qDebug() << "Running" << application << "with args" << args.join(' ');
|
||||||
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
||||||
if (!isSandbox()) {
|
if(!isFlatpak())
|
||||||
// FIXME: the pid here is fake. So if something depends on it, it will likely misbehave
|
{
|
||||||
return IndirectOpen([&]() { return QProcess::startDetached(application, args, workingDirectory); }, pid);
|
// FIXME: the pid here is fake. So if something depends on it, it will likely misbehave
|
||||||
} else {
|
return IndirectOpen([&]()
|
||||||
|
{
|
||||||
|
return QProcess::startDetached(application, args, workingDirectory);
|
||||||
|
}, pid);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
return QProcess::startDetached(application, args, workingDirectory, pid);
|
return QProcess::startDetached(application, args, workingDirectory, pid);
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
@ -161,14 +194,20 @@ bool run(const QString& application, const QStringList& args, const QString& wor
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
bool openUrl(const QUrl& url)
|
bool openUrl(const QUrl &url)
|
||||||
{
|
{
|
||||||
qDebug() << "Opening URL" << url.toString();
|
qDebug() << "Opening URL" << url.toString();
|
||||||
auto f = [&]() { return QDesktopServices::openUrl(url); };
|
auto f = [&]()
|
||||||
|
{
|
||||||
|
return QDesktopServices::openUrl(url);
|
||||||
|
};
|
||||||
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
||||||
if (!isSandbox()) {
|
if(!isFlatpak())
|
||||||
|
{
|
||||||
return IndirectOpen(f);
|
return IndirectOpen(f);
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
return f();
|
return f();
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
@ -185,18 +224,4 @@ bool isFlatpak()
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
bool isSnap()
|
|
||||||
{
|
|
||||||
#ifdef Q_OS_LINUX
|
|
||||||
return getenv("SNAP");
|
|
||||||
#else
|
|
||||||
return false;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool isSandbox()
|
|
||||||
{
|
|
||||||
return isSnap() || isFlatpak();
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace DesktopServices
|
|
||||||
|
@ -1,50 +1,38 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QString>
|
|
||||||
#include <QUrl>
|
#include <QUrl>
|
||||||
|
#include <QString>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This wraps around QDesktopServices and adds workarounds where needed
|
* This wraps around QDesktopServices and adds workarounds where needed
|
||||||
* Use this instead of QDesktopServices!
|
* Use this instead of QDesktopServices!
|
||||||
*/
|
*/
|
||||||
namespace DesktopServices {
|
namespace DesktopServices
|
||||||
/**
|
{
|
||||||
* Open a file in whatever application is applicable
|
/**
|
||||||
*/
|
* Open a file in whatever application is applicable
|
||||||
bool openFile(const QString& path);
|
*/
|
||||||
|
bool openFile(const QString &path);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Open a file in the specified application
|
* Open a file in the specified application
|
||||||
*/
|
*/
|
||||||
bool openFile(const QString& application, const QString& path, const QString& workingDirectory = QString(), qint64* pid = 0);
|
bool openFile(const QString &application, const QString &path, const QString & workingDirectory = QString(), qint64 *pid = 0);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Run an application
|
* Run an application
|
||||||
*/
|
*/
|
||||||
bool run(const QString& application, const QStringList& args, const QString& workingDirectory = QString(), qint64* pid = 0);
|
bool run(const QString &application,const QStringList &args, const QString & workingDirectory = QString(), qint64 *pid = 0);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Open a directory
|
* Open a directory
|
||||||
*/
|
*/
|
||||||
bool openDirectory(const QString& path, bool ensureExists = false);
|
bool openDirectory(const QString &path, bool ensureExists = false);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Open the URL, most likely in a browser. Maybe.
|
* Open the URL, most likely in a browser. Maybe.
|
||||||
*/
|
*/
|
||||||
bool openUrl(const QUrl& url);
|
bool openUrl(const QUrl &url);
|
||||||
|
|
||||||
/**
|
bool isFlatpak();
|
||||||
* Determine whether the launcher is running in a Flatpak environment
|
}
|
||||||
*/
|
|
||||||
bool isFlatpak();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Determine whether the launcher is running in a Snap environment
|
|
||||||
*/
|
|
||||||
bool isSnap();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Determine whether the launcher is running in a sandboxed (Flatpak or Snap) environment
|
|
||||||
*/
|
|
||||||
bool isSandbox();
|
|
||||||
} // namespace DesktopServices
|
|
||||||
|
@ -2,18 +2,31 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QDebug>
|
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
#include <QDebug>
|
||||||
#include <exception>
|
#include <exception>
|
||||||
|
|
||||||
class Exception : public std::exception {
|
class Exception : public std::exception
|
||||||
public:
|
{
|
||||||
Exception(const QString& message) : std::exception(), m_message(message) { qCritical() << "Exception:" << message; }
|
public:
|
||||||
Exception(const Exception& other) : std::exception(), m_message(other.cause()) {}
|
Exception(const QString &message) : std::exception(), m_message(message)
|
||||||
|
{
|
||||||
|
qCritical() << "Exception:" << message;
|
||||||
|
}
|
||||||
|
Exception(const Exception &other)
|
||||||
|
: std::exception(), m_message(other.cause())
|
||||||
|
{
|
||||||
|
}
|
||||||
virtual ~Exception() noexcept {}
|
virtual ~Exception() noexcept {}
|
||||||
const char* what() const noexcept { return m_message.toLatin1().constData(); }
|
const char *what() const noexcept
|
||||||
QString cause() const { return m_message; }
|
{
|
||||||
|
return m_message.toLatin1().constData();
|
||||||
|
}
|
||||||
|
QString cause() const
|
||||||
|
{
|
||||||
|
return m_message;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QString m_message;
|
QString m_message;
|
||||||
};
|
};
|
||||||
|
@ -4,24 +4,31 @@
|
|||||||
template <typename T>
|
template <typename T>
|
||||||
inline void clamp(T& current, T min, T max)
|
inline void clamp(T& current, T min, T max)
|
||||||
{
|
{
|
||||||
if (current < min) {
|
if (current < min)
|
||||||
|
{
|
||||||
current = min;
|
current = min;
|
||||||
} else if (current > max) {
|
}
|
||||||
|
else if(current > max)
|
||||||
|
{
|
||||||
current = max;
|
current = max;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// List of numbers from min to max. Next is exponent times bigger than previous.
|
// List of numbers from min to max. Next is exponent times bigger than previous.
|
||||||
|
|
||||||
class ExponentialSeries {
|
class ExponentialSeries
|
||||||
public:
|
{
|
||||||
|
public:
|
||||||
ExponentialSeries(unsigned min, unsigned max, unsigned exponent = 2)
|
ExponentialSeries(unsigned min, unsigned max, unsigned exponent = 2)
|
||||||
{
|
{
|
||||||
m_current = m_min = min;
|
m_current = m_min = min;
|
||||||
m_max = max;
|
m_max = max;
|
||||||
m_exponent = exponent;
|
m_exponent = exponent;
|
||||||
}
|
}
|
||||||
void reset() { m_current = m_min; }
|
void reset()
|
||||||
|
{
|
||||||
|
m_current = m_min;
|
||||||
|
}
|
||||||
unsigned operator()()
|
unsigned operator()()
|
||||||
{
|
{
|
||||||
unsigned retval = m_current;
|
unsigned retval = m_current;
|
||||||
|
@ -267,7 +267,10 @@ bool FileIgnoreProxy::filterAcceptsRow(int sourceRow, const QModelIndex& sourceP
|
|||||||
|
|
||||||
bool FileIgnoreProxy::ignoreFile(QFileInfo fileInfo) const
|
bool FileIgnoreProxy::ignoreFile(QFileInfo fileInfo) const
|
||||||
{
|
{
|
||||||
return m_ignoreFiles.contains(fileInfo.fileName()) || m_ignoreFilePaths.covers(relPath(fileInfo.absoluteFilePath()));
|
auto fileName = fileInfo.fileName();
|
||||||
|
auto path = relPath(fileInfo.absoluteFilePath());
|
||||||
|
return std::any_of(m_ignoreFiles.cbegin(), m_ignoreFiles.cend(), [fileName](auto iFileName) { return fileName == iFileName; }) ||
|
||||||
|
m_ignoreFilePaths.covers(path);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool FileIgnoreProxy::filterFile(const QString& fileName) const
|
bool FileIgnoreProxy::filterFile(const QString& fileName) const
|
||||||
|
@ -123,35 +123,26 @@ namespace fs = ghc::filesystem;
|
|||||||
|
|
||||||
#if defined(__MINGW32__)
|
#if defined(__MINGW32__)
|
||||||
|
|
||||||
struct _DUPLICATE_EXTENTS_DATA {
|
typedef struct _DUPLICATE_EXTENTS_DATA {
|
||||||
HANDLE FileHandle;
|
HANDLE FileHandle;
|
||||||
LARGE_INTEGER SourceFileOffset;
|
LARGE_INTEGER SourceFileOffset;
|
||||||
LARGE_INTEGER TargetFileOffset;
|
LARGE_INTEGER TargetFileOffset;
|
||||||
LARGE_INTEGER ByteCount;
|
LARGE_INTEGER ByteCount;
|
||||||
};
|
} DUPLICATE_EXTENTS_DATA, *PDUPLICATE_EXTENTS_DATA;
|
||||||
|
|
||||||
using DUPLICATE_EXTENTS_DATA = _DUPLICATE_EXTENTS_DATA;
|
typedef struct _FSCTL_GET_INTEGRITY_INFORMATION_BUFFER {
|
||||||
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 ChecksumAlgorithm; // Checksum algorithm. e.g. CHECKSUM_TYPE_UNCHANGED, CHECKSUM_TYPE_NONE, CHECKSUM_TYPE_CRC32
|
||||||
WORD Reserved; // Must be 0
|
WORD Reserved; // Must be 0
|
||||||
DWORD Flags; // FSCTL_INTEGRITY_FLAG_xxx
|
DWORD Flags; // FSCTL_INTEGRITY_FLAG_xxx
|
||||||
DWORD ChecksumChunkSizeInBytes;
|
DWORD ChecksumChunkSizeInBytes;
|
||||||
DWORD ClusterSizeInBytes;
|
DWORD ClusterSizeInBytes;
|
||||||
};
|
} FSCTL_GET_INTEGRITY_INFORMATION_BUFFER, *PFSCTL_GET_INTEGRITY_INFORMATION_BUFFER;
|
||||||
|
|
||||||
using FSCTL_GET_INTEGRITY_INFORMATION_BUFFER = _FSCTL_GET_INTEGRITY_INFORMATION_BUFFER;
|
typedef struct _FSCTL_SET_INTEGRITY_INFORMATION_BUFFER {
|
||||||
using PFSCTL_GET_INTEGRITY_INFORMATION_BUFFER = _FSCTL_GET_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 ChecksumAlgorithm; // Checksum algorithm. e.g. CHECKSUM_TYPE_UNCHANGED, CHECKSUM_TYPE_NONE, CHECKSUM_TYPE_CRC32
|
||||||
WORD Reserved; // Must be 0
|
WORD Reserved; // Must be 0
|
||||||
DWORD Flags; // FSCTL_INTEGRITY_FLAG_xxx
|
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
|
#endif
|
||||||
|
|
||||||
@ -203,40 +194,6 @@ 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)
|
QByteArray read(const QString& filename)
|
||||||
{
|
{
|
||||||
QFile file(filename);
|
QFile file(filename);
|
||||||
@ -281,28 +238,6 @@ bool ensureFolderPathExists(QString foldernamepath)
|
|||||||
return success;
|
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
|
* @brief Copies a directory and it's contents from src to dest
|
||||||
* @param offset subdirectory form src to copy to dest
|
* @param offset subdirectory form src to copy to dest
|
||||||
@ -330,9 +265,6 @@ bool copy::operator()(const QString& offset, bool dryRun)
|
|||||||
if (!m_followSymlinks)
|
if (!m_followSymlinks)
|
||||||
opt |= copy_opts::copy_symlinks;
|
opt |= copy_opts::copy_symlinks;
|
||||||
|
|
||||||
if (m_overwrite)
|
|
||||||
opt |= copy_opts::overwrite_existing;
|
|
||||||
|
|
||||||
// Function that'll do the actual copying
|
// Function that'll do the actual copying
|
||||||
auto copy_file = [&](QString src_path, QString relative_dst_path) {
|
auto copy_file = [&](QString src_path, QString relative_dst_path) {
|
||||||
if (m_matcher && (m_matcher->matches(relative_dst_path) != m_whitelist))
|
if (m_matcher && (m_matcher->matches(relative_dst_path) != m_whitelist))
|
||||||
@ -341,9 +273,6 @@ bool copy::operator()(const QString& offset, bool dryRun)
|
|||||||
auto dst_path = PathCombine(dst, relative_dst_path);
|
auto dst_path = PathCombine(dst, relative_dst_path);
|
||||||
if (!dryRun) {
|
if (!dryRun) {
|
||||||
ensureFilePathExists(dst_path);
|
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);
|
fs::copy(StringUtils::toStdString(src_path), StringUtils::toStdString(dst_path), opt, err);
|
||||||
}
|
}
|
||||||
if (err) {
|
if (err) {
|
||||||
@ -849,44 +778,9 @@ bool createShortcut(QString destination, QString target, QStringList args, QStri
|
|||||||
destination = PathCombine(getDesktopDir(), RemoveInvalidFilenameChars(name));
|
destination = PathCombine(getDesktopDir(), RemoveInvalidFilenameChars(name));
|
||||||
}
|
}
|
||||||
#if defined(Q_OS_MACOS)
|
#if defined(Q_OS_MACOS)
|
||||||
// Create the Application
|
destination += ".command";
|
||||||
QDir applicationDirectory =
|
|
||||||
QStandardPaths::writableLocation(QStandardPaths::ApplicationsLocation) + "/" + BuildConfig.LAUNCHER_NAME + " Instances/";
|
|
||||||
|
|
||||||
if (!applicationDirectory.mkpath(".")) {
|
QFile f(destination);
|
||||||
qWarning() << "Couldn't create application directory";
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
QDir application = applicationDirectory.path() + "/" + name + ".app/";
|
|
||||||
|
|
||||||
if (application.exists()) {
|
|
||||||
qWarning() << "Application already exists!";
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!application.mkpath(".")) {
|
|
||||||
qWarning() << "Couldn't create application";
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
QDir content = application.path() + "/Contents/";
|
|
||||||
QDir resources = content.path() + "/Resources/";
|
|
||||||
QDir binaryDir = content.path() + "/MacOS/";
|
|
||||||
QFile info = content.path() + "/Info.plist";
|
|
||||||
|
|
||||||
if (!(content.mkpath(".") && resources.mkpath(".") && binaryDir.mkpath("."))) {
|
|
||||||
qWarning() << "Couldn't create directories within application";
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
info.open(QIODevice::WriteOnly | QIODevice::Text);
|
|
||||||
|
|
||||||
QFile(icon).rename(resources.path() + "/Icon.icns");
|
|
||||||
|
|
||||||
// Create the Command file
|
|
||||||
QString exec = binaryDir.path() + "/Run.command";
|
|
||||||
|
|
||||||
QFile f(exec);
|
|
||||||
f.open(QIODevice::WriteOnly | QIODevice::Text);
|
f.open(QIODevice::WriteOnly | QIODevice::Text);
|
||||||
QTextStream stream(&f);
|
QTextStream stream(&f);
|
||||||
|
|
||||||
@ -903,30 +797,6 @@ bool createShortcut(QString destination, QString target, QStringList args, QStri
|
|||||||
|
|
||||||
f.setPermissions(f.permissions() | QFileDevice::ExeOwner | QFileDevice::ExeGroup | QFileDevice::ExeOther);
|
f.setPermissions(f.permissions() | QFileDevice::ExeOwner | QFileDevice::ExeGroup | QFileDevice::ExeOther);
|
||||||
|
|
||||||
// Generate the Info.plist
|
|
||||||
QTextStream infoStream(&info);
|
|
||||||
infoStream << "<?xml version=\"1.0\" encoding=\"UTF-8\"?> \n"
|
|
||||||
"<!DOCTYPE plist PUBLIC \"-//Apple Computer//DTD PLIST 1.0//EN\" "
|
|
||||||
"\"http://www.apple.com/DTDs/PropertyList-1.0.dtd\">"
|
|
||||||
"<plist version=\"1.0\">\n"
|
|
||||||
"<dict>\n"
|
|
||||||
" <key>CFBundleExecutable</key>\n"
|
|
||||||
" <string>Run.command</string>\n" // The path to the executable
|
|
||||||
" <key>CFBundleIconFile</key>\n"
|
|
||||||
" <string>Icon.icns</string>\n"
|
|
||||||
" <key>CFBundleName</key>\n"
|
|
||||||
" <string>"
|
|
||||||
<< name
|
|
||||||
<< "</string>\n" // Name of the application
|
|
||||||
" <key>CFBundlePackageType</key>\n"
|
|
||||||
" <string>APPL</string>\n"
|
|
||||||
" <key>CFBundleShortVersionString</key>\n"
|
|
||||||
" <string>1.0</string>\n"
|
|
||||||
" <key>CFBundleVersion</key>\n"
|
|
||||||
" <string>1.0</string>\n"
|
|
||||||
"</dict>\n"
|
|
||||||
"</plist>";
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
#elif defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD) || defined(Q_OS_OPENBSD)
|
#elif defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD) || defined(Q_OS_OPENBSD)
|
||||||
if (!destination.endsWith(".desktop")) // in case of isFlatpak destination is already populated
|
if (!destination.endsWith(".desktop")) // in case of isFlatpak destination is already populated
|
||||||
|
@ -43,10 +43,10 @@
|
|||||||
#include <system_error>
|
#include <system_error>
|
||||||
|
|
||||||
#include <QDir>
|
#include <QDir>
|
||||||
|
#include <QPair>
|
||||||
#include <QFlags>
|
#include <QFlags>
|
||||||
#include <QLocalServer>
|
#include <QLocalServer>
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QPair>
|
|
||||||
#include <QThread>
|
#include <QThread>
|
||||||
|
|
||||||
namespace FS {
|
namespace FS {
|
||||||
@ -61,16 +61,6 @@ class FileSystemException : public ::Exception {
|
|||||||
*/
|
*/
|
||||||
void write(const QString& filename, const QByteArray& data);
|
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\
|
* read data from a file safely\
|
||||||
*/
|
*/
|
||||||
@ -119,16 +109,11 @@ class copy : public QObject {
|
|||||||
m_whitelist = whitelist;
|
m_whitelist = whitelist;
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
copy& overwrite(const bool overwrite)
|
|
||||||
{
|
|
||||||
m_overwrite = overwrite;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator()(bool dryRun = false) { return operator()(QString(), dryRun); }
|
bool operator()(bool dryRun = false) { return operator()(QString(), dryRun); }
|
||||||
|
|
||||||
qsizetype totalCopied() { return m_copied; }
|
int totalCopied() { return m_copied; }
|
||||||
qsizetype totalFailed() { return m_failedPaths.length(); }
|
int totalFailed() { return m_failedPaths.length(); }
|
||||||
QStringList failed() { return m_failedPaths; }
|
QStringList failed() { return m_failedPaths; }
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
@ -143,10 +128,9 @@ class copy : public QObject {
|
|||||||
bool m_followSymlinks = true;
|
bool m_followSymlinks = true;
|
||||||
const IPathMatcher* m_matcher = nullptr;
|
const IPathMatcher* m_matcher = nullptr;
|
||||||
bool m_whitelist = false;
|
bool m_whitelist = false;
|
||||||
bool m_overwrite = false;
|
|
||||||
QDir m_src;
|
QDir m_src;
|
||||||
QDir m_dst;
|
QDir m_dst;
|
||||||
qsizetype m_copied;
|
int m_copied;
|
||||||
QStringList m_failedPaths;
|
QStringList m_failedPaths;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -381,24 +365,25 @@ enum class FilesystemType {
|
|||||||
* QMap is ordered
|
* QMap is ordered
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
static const QMap<FilesystemType, QStringList> s_filesystem_type_names = { { FilesystemType::FAT, { "FAT" } },
|
static const QMap<FilesystemType, QStringList> s_filesystem_type_names = {
|
||||||
{ FilesystemType::NTFS, { "NTFS" } },
|
{FilesystemType::FAT, { "FAT" }},
|
||||||
{ FilesystemType::REFS, { "REFS" } },
|
{FilesystemType::NTFS, { "NTFS" }},
|
||||||
{ FilesystemType::EXT_2_OLD, { "EXT_2_OLD", "EXT2_OLD" } },
|
{FilesystemType::REFS, { "REFS" }},
|
||||||
{ FilesystemType::EXT_2_3_4,
|
{FilesystemType::EXT_2_OLD, { "EXT_2_OLD", "EXT2_OLD" }},
|
||||||
{ "EXT2/3/4", "EXT_2_3_4", "EXT2", "EXT3", "EXT4" } },
|
{FilesystemType::EXT_2_3_4, { "EXT2/3/4", "EXT_2_3_4", "EXT2", "EXT3", "EXT4" }},
|
||||||
{ FilesystemType::EXT, { "EXT" } },
|
{FilesystemType::EXT, { "EXT" }},
|
||||||
{ FilesystemType::XFS, { "XFS" } },
|
{FilesystemType::XFS, { "XFS" }},
|
||||||
{ FilesystemType::BTRFS, { "BTRFS" } },
|
{FilesystemType::BTRFS, { "BTRFS" }},
|
||||||
{ FilesystemType::NFS, { "NFS" } },
|
{FilesystemType::NFS, { "NFS" }},
|
||||||
{ FilesystemType::ZFS, { "ZFS" } },
|
{FilesystemType::ZFS, { "ZFS" }},
|
||||||
{ FilesystemType::APFS, { "APFS" } },
|
{FilesystemType::APFS, { "APFS" }},
|
||||||
{ FilesystemType::HFS, { "HFS" } },
|
{FilesystemType::HFS, { "HFS" }},
|
||||||
{ FilesystemType::HFSPLUS, { "HFSPLUS" } },
|
{FilesystemType::HFSPLUS, { "HFSPLUS" }},
|
||||||
{ FilesystemType::HFSX, { "HFSX" } },
|
{FilesystemType::HFSX, { "HFSX" }},
|
||||||
{ FilesystemType::FUSEBLK, { "FUSEBLK" } },
|
{FilesystemType::FUSEBLK, { "FUSEBLK" }},
|
||||||
{ FilesystemType::F2FS, { "F2FS" } },
|
{FilesystemType::F2FS, { "F2FS" }},
|
||||||
{ FilesystemType::UNKNOWN, { "UNKNOWN" } } };
|
{FilesystemType::UNKNOWN, { "UNKNOWN" }}
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Get the string name of Filesystem enum object
|
* @brief Get the string name of Filesystem enum object
|
||||||
@ -490,8 +475,8 @@ class clone : public QObject {
|
|||||||
|
|
||||||
bool operator()(bool dryRun = false) { return operator()(QString(), dryRun); }
|
bool operator()(bool dryRun = false) { return operator()(QString(), dryRun); }
|
||||||
|
|
||||||
qsizetype totalCloned() { return m_cloned; }
|
int totalCloned() { return m_cloned; }
|
||||||
qsizetype totalFailed() { return m_failedClones.length(); }
|
int totalFailed() { return m_failedClones.length(); }
|
||||||
|
|
||||||
QList<QPair<QString, QString>> failed() { return m_failedClones; }
|
QList<QPair<QString, QString>> failed() { return m_failedClones; }
|
||||||
|
|
||||||
@ -507,7 +492,7 @@ class clone : public QObject {
|
|||||||
bool m_whitelist = false;
|
bool m_whitelist = false;
|
||||||
QDir m_src;
|
QDir m_src;
|
||||||
QDir m_dst;
|
QDir m_dst;
|
||||||
qsizetype m_cloned;
|
int m_cloned;
|
||||||
QList<QPair<QString, QString>> m_failedClones;
|
QList<QPair<QString, QString>> m_failedClones;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1,33 +1,28 @@
|
|||||||
#include "Filter.h"
|
#include "Filter.h"
|
||||||
|
|
||||||
Filter::~Filter() {}
|
Filter::~Filter(){}
|
||||||
|
|
||||||
ContainsFilter::ContainsFilter(const QString& pattern) : pattern(pattern) {}
|
ContainsFilter::ContainsFilter(const QString& pattern) : pattern(pattern){}
|
||||||
ContainsFilter::~ContainsFilter() {}
|
ContainsFilter::~ContainsFilter(){}
|
||||||
bool ContainsFilter::accepts(const QString& value)
|
bool ContainsFilter::accepts(const QString& value)
|
||||||
{
|
{
|
||||||
return value.contains(pattern);
|
return value.contains(pattern);
|
||||||
}
|
}
|
||||||
|
|
||||||
ExactFilter::ExactFilter(const QString& pattern) : pattern(pattern) {}
|
ExactFilter::ExactFilter(const QString& pattern) : pattern(pattern){}
|
||||||
ExactFilter::~ExactFilter() {}
|
ExactFilter::~ExactFilter(){}
|
||||||
bool ExactFilter::accepts(const QString& value)
|
bool ExactFilter::accepts(const QString& value)
|
||||||
{
|
{
|
||||||
return value == pattern;
|
return value == pattern;
|
||||||
}
|
}
|
||||||
|
|
||||||
ExactIfPresentFilter::ExactIfPresentFilter(const QString& pattern) : pattern(pattern) {}
|
RegexpFilter::RegexpFilter(const QString& regexp, bool invert)
|
||||||
bool ExactIfPresentFilter::accepts(const QString& value)
|
:invert(invert)
|
||||||
{
|
|
||||||
return value.isEmpty() || value == pattern;
|
|
||||||
}
|
|
||||||
|
|
||||||
RegexpFilter::RegexpFilter(const QString& regexp, bool invert) : invert(invert)
|
|
||||||
{
|
{
|
||||||
pattern.setPattern(regexp);
|
pattern.setPattern(regexp);
|
||||||
pattern.optimize();
|
pattern.optimize();
|
||||||
}
|
}
|
||||||
RegexpFilter::~RegexpFilter() {}
|
RegexpFilter::~RegexpFilter(){}
|
||||||
bool RegexpFilter::accepts(const QString& value)
|
bool RegexpFilter::accepts(const QString& value)
|
||||||
{
|
{
|
||||||
auto match = pattern.match(value);
|
auto match = pattern.match(value);
|
||||||
|
@ -1,51 +1,42 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QRegularExpression>
|
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
#include <QRegularExpression>
|
||||||
|
|
||||||
class Filter {
|
class Filter
|
||||||
public:
|
{
|
||||||
|
public:
|
||||||
virtual ~Filter();
|
virtual ~Filter();
|
||||||
virtual bool accepts(const QString& value) = 0;
|
virtual bool accepts(const QString & value) = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ContainsFilter : public Filter {
|
class ContainsFilter: public Filter
|
||||||
public:
|
{
|
||||||
ContainsFilter(const QString& pattern);
|
public:
|
||||||
|
ContainsFilter(const QString &pattern);
|
||||||
virtual ~ContainsFilter();
|
virtual ~ContainsFilter();
|
||||||
bool accepts(const QString& value) override;
|
bool accepts(const QString & value) override;
|
||||||
|
private:
|
||||||
private:
|
|
||||||
QString pattern;
|
QString pattern;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ExactFilter : public Filter {
|
class ExactFilter: public Filter
|
||||||
public:
|
{
|
||||||
ExactFilter(const QString& pattern);
|
public:
|
||||||
|
ExactFilter(const QString &pattern);
|
||||||
virtual ~ExactFilter();
|
virtual ~ExactFilter();
|
||||||
bool accepts(const QString& value) override;
|
bool accepts(const QString & value) override;
|
||||||
|
private:
|
||||||
private:
|
|
||||||
QString pattern;
|
QString pattern;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ExactIfPresentFilter : public Filter {
|
class RegexpFilter: public Filter
|
||||||
public:
|
{
|
||||||
ExactIfPresentFilter(const QString& pattern);
|
public:
|
||||||
~ExactIfPresentFilter() override = default;
|
RegexpFilter(const QString ®exp, bool invert);
|
||||||
bool accepts(const QString& value) override;
|
|
||||||
|
|
||||||
private:
|
|
||||||
QString pattern;
|
|
||||||
};
|
|
||||||
|
|
||||||
class RegexpFilter : public Filter {
|
|
||||||
public:
|
|
||||||
RegexpFilter(const QString& regexp, bool invert);
|
|
||||||
virtual ~RegexpFilter();
|
virtual ~RegexpFilter();
|
||||||
bool accepts(const QString& value) override;
|
bool accepts(const QString & value) override;
|
||||||
|
private:
|
||||||
private:
|
|
||||||
QRegularExpression pattern;
|
QRegularExpression pattern;
|
||||||
bool invert = false;
|
bool invert = false;
|
||||||
};
|
};
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
@ -37,9 +37,10 @@
|
|||||||
#include <zlib.h>
|
#include <zlib.h>
|
||||||
#include <QByteArray>
|
#include <QByteArray>
|
||||||
|
|
||||||
bool GZip::unzip(const QByteArray& compressedBytes, QByteArray& uncompressedBytes)
|
bool GZip::unzip(const QByteArray &compressedBytes, QByteArray &uncompressedBytes)
|
||||||
{
|
{
|
||||||
if (compressedBytes.size() == 0) {
|
if (compressedBytes.size() == 0)
|
||||||
|
{
|
||||||
uncompressedBytes = compressedBytes;
|
uncompressedBytes = compressedBytes;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -50,37 +51,42 @@ bool GZip::unzip(const QByteArray& compressedBytes, QByteArray& uncompressedByte
|
|||||||
|
|
||||||
z_stream strm;
|
z_stream strm;
|
||||||
memset(&strm, 0, sizeof(strm));
|
memset(&strm, 0, sizeof(strm));
|
||||||
strm.next_in = (Bytef*)compressedBytes.data();
|
strm.next_in = (Bytef *)compressedBytes.data();
|
||||||
strm.avail_in = compressedBytes.size();
|
strm.avail_in = compressedBytes.size();
|
||||||
|
|
||||||
bool done = false;
|
bool done = false;
|
||||||
|
|
||||||
if (inflateInit2(&strm, (16 + MAX_WBITS)) != Z_OK) {
|
if (inflateInit2(&strm, (16 + MAX_WBITS)) != Z_OK)
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
int err = Z_OK;
|
int err = Z_OK;
|
||||||
|
|
||||||
while (!done) {
|
while (!done)
|
||||||
|
{
|
||||||
// If our output buffer is too small
|
// If our output buffer is too small
|
||||||
if (strm.total_out >= uncompLength) {
|
if (strm.total_out >= uncompLength)
|
||||||
|
{
|
||||||
uncompressedBytes.resize(uncompLength * 2);
|
uncompressedBytes.resize(uncompLength * 2);
|
||||||
uncompLength *= 2;
|
uncompLength *= 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
strm.next_out = reinterpret_cast<Bytef*>((uncompressedBytes.data() + strm.total_out));
|
strm.next_out = reinterpret_cast<Bytef *>((uncompressedBytes.data() + strm.total_out));
|
||||||
strm.avail_out = uncompLength - strm.total_out;
|
strm.avail_out = uncompLength - strm.total_out;
|
||||||
|
|
||||||
// Inflate another chunk.
|
// Inflate another chunk.
|
||||||
err = inflate(&strm, Z_SYNC_FLUSH);
|
err = inflate(&strm, Z_SYNC_FLUSH);
|
||||||
if (err == Z_STREAM_END)
|
if (err == Z_STREAM_END)
|
||||||
done = true;
|
done = true;
|
||||||
else if (err != Z_OK) {
|
else if (err != Z_OK)
|
||||||
|
{
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (inflateEnd(&strm) != Z_OK || !done) {
|
if (inflateEnd(&strm) != Z_OK || !done)
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -88,9 +94,10 @@ bool GZip::unzip(const QByteArray& compressedBytes, QByteArray& uncompressedByte
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool GZip::zip(const QByteArray& uncompressedBytes, QByteArray& compressedBytes)
|
bool GZip::zip(const QByteArray &uncompressedBytes, QByteArray &compressedBytes)
|
||||||
{
|
{
|
||||||
if (uncompressedBytes.size() == 0) {
|
if (uncompressedBytes.size() == 0)
|
||||||
|
{
|
||||||
compressedBytes = uncompressedBytes;
|
compressedBytes = uncompressedBytes;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -102,7 +109,8 @@ bool GZip::zip(const QByteArray& uncompressedBytes, QByteArray& compressedBytes)
|
|||||||
z_stream zs;
|
z_stream zs;
|
||||||
memset(&zs, 0, sizeof(zs));
|
memset(&zs, 0, sizeof(zs));
|
||||||
|
|
||||||
if (deflateInit2(&zs, Z_DEFAULT_COMPRESSION, Z_DEFLATED, (16 + MAX_WBITS), 8, Z_DEFAULT_STRATEGY) != Z_OK) {
|
if (deflateInit2(&zs, Z_DEFAULT_COMPRESSION, Z_DEFLATED, (16 + MAX_WBITS), 8, Z_DEFAULT_STRATEGY) != Z_OK)
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -114,9 +122,11 @@ bool GZip::zip(const QByteArray& uncompressedBytes, QByteArray& compressedBytes)
|
|||||||
|
|
||||||
unsigned offset = 0;
|
unsigned offset = 0;
|
||||||
unsigned temp = 0;
|
unsigned temp = 0;
|
||||||
do {
|
do
|
||||||
|
{
|
||||||
auto remaining = compressedBytes.size() - offset;
|
auto remaining = compressedBytes.size() - offset;
|
||||||
if (remaining < 1) {
|
if(remaining < 1)
|
||||||
|
{
|
||||||
compressedBytes.resize(compressedBytes.size() * 2);
|
compressedBytes.resize(compressedBytes.size() * 2);
|
||||||
}
|
}
|
||||||
zs.next_out = reinterpret_cast<Bytef*>((compressedBytes.data() + offset));
|
zs.next_out = reinterpret_cast<Bytef*>((compressedBytes.data() + offset));
|
||||||
@ -127,11 +137,13 @@ bool GZip::zip(const QByteArray& uncompressedBytes, QByteArray& compressedBytes)
|
|||||||
|
|
||||||
compressedBytes.resize(offset);
|
compressedBytes.resize(offset);
|
||||||
|
|
||||||
if (deflateEnd(&zs) != Z_OK) {
|
if (deflateEnd(&zs) != Z_OK)
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret != Z_STREAM_END) {
|
if (ret != Z_STREAM_END)
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <QByteArray>
|
#include <QByteArray>
|
||||||
|
|
||||||
class GZip {
|
class GZip
|
||||||
public:
|
{
|
||||||
static bool unzip(const QByteArray& compressedBytes, QByteArray& uncompressedBytes);
|
public:
|
||||||
static bool zip(const QByteArray& uncompressedBytes, QByteArray& compressedBytes);
|
static bool unzip(const QByteArray &compressedBytes, QByteArray &uncompressedBytes);
|
||||||
|
static bool zip(const QByteArray &uncompressedBytes, QByteArray &compressedBytes);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -6,10 +6,17 @@
|
|||||||
|
|
||||||
bool InstanceCopyPrefs::allTrue() const
|
bool InstanceCopyPrefs::allTrue() const
|
||||||
{
|
{
|
||||||
return copySaves && keepPlaytime && copyGameOptions && copyResourcePacks && copyShaderPacks && copyServers && copyMods &&
|
return copySaves &&
|
||||||
copyScreenshots;
|
keepPlaytime &&
|
||||||
|
copyGameOptions &&
|
||||||
|
copyResourcePacks &&
|
||||||
|
copyShaderPacks &&
|
||||||
|
copyServers &&
|
||||||
|
copyMods &&
|
||||||
|
copyScreenshots;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Returns a single RegEx string of the selected folders/files to filter out (ex: ".minecraft/saves|.minecraft/server.dat")
|
// Returns a single RegEx string of the selected folders/files to filter out (ex: ".minecraft/saves|.minecraft/server.dat")
|
||||||
QString InstanceCopyPrefs::getSelectedFiltersAsRegex() const
|
QString InstanceCopyPrefs::getSelectedFiltersAsRegex() const
|
||||||
{
|
{
|
||||||
@ -19,30 +26,25 @@ QString InstanceCopyPrefs::getSelectedFiltersAsRegex(const QStringList& addition
|
|||||||
{
|
{
|
||||||
QStringList filters;
|
QStringList filters;
|
||||||
|
|
||||||
if (!copySaves)
|
if(!copySaves)
|
||||||
filters << "saves";
|
filters << "saves";
|
||||||
|
|
||||||
if (!copyGameOptions)
|
if(!copyGameOptions)
|
||||||
filters << "options.txt";
|
filters << "options.txt";
|
||||||
|
|
||||||
if (!copyResourcePacks)
|
if(!copyResourcePacks)
|
||||||
filters << "resourcepacks"
|
filters << "resourcepacks" << "texturepacks";
|
||||||
<< "texturepacks";
|
|
||||||
|
|
||||||
if (!copyShaderPacks)
|
if(!copyShaderPacks)
|
||||||
filters << "shaderpacks";
|
filters << "shaderpacks";
|
||||||
|
|
||||||
if (!copyServers)
|
if(!copyServers)
|
||||||
filters << "servers.dat"
|
filters << "servers.dat" << "servers.dat_old" << "server-resource-packs";
|
||||||
<< "servers.dat_old"
|
|
||||||
<< "server-resource-packs";
|
|
||||||
|
|
||||||
if (!copyMods)
|
if(!copyMods)
|
||||||
filters << "coremods"
|
filters << "coremods" << "mods" << "config";
|
||||||
<< "mods"
|
|
||||||
<< "config";
|
|
||||||
|
|
||||||
if (!copyScreenshots)
|
if(!copyScreenshots)
|
||||||
filters << "screenshots";
|
filters << "screenshots";
|
||||||
|
|
||||||
for (auto filter : additionalFilters) {
|
for (auto filter : additionalFilters) {
|
||||||
|
@ -40,7 +40,7 @@ struct InstanceCopyPrefs {
|
|||||||
void enableDontLinkSaves(bool b);
|
void enableDontLinkSaves(bool b);
|
||||||
void enableUseClone(bool b);
|
void enableUseClone(bool b);
|
||||||
|
|
||||||
protected: // data
|
protected: // data
|
||||||
bool copySaves = true;
|
bool copySaves = true;
|
||||||
bool keepPlaytime = true;
|
bool keepPlaytime = true;
|
||||||
bool copyGameOptions = true;
|
bool copyGameOptions = true;
|
||||||
|
@ -156,9 +156,8 @@ void InstanceCopyTask::copyFinished()
|
|||||||
allowed_symlinks.append(m_origInstance->gameRoot().toUtf8());
|
allowed_symlinks.append(m_origInstance->gameRoot().toUtf8());
|
||||||
allowed_symlinks.append("\n");
|
allowed_symlinks.append("\n");
|
||||||
if (allowed_symlinks_file.isSymLink())
|
if (allowed_symlinks_file.isSymLink())
|
||||||
FS::deletePath(
|
FS::deletePath(allowed_symlinks_file
|
||||||
allowed_symlinks_file
|
.filePath()); // we dont want to modify the original. also make sure the resulting file is not itself a link.
|
||||||
.filePath()); // we dont want to modify the original. also make sure the resulting file is not itself a link.
|
|
||||||
|
|
||||||
FS::write(allowed_symlinks_file.filePath(), allowed_symlinks);
|
FS::write(allowed_symlinks_file.filePath(), allowed_symlinks);
|
||||||
}
|
}
|
||||||
|
@ -11,18 +11,19 @@
|
|||||||
#include "settings/SettingsObject.h"
|
#include "settings/SettingsObject.h"
|
||||||
#include "tasks/Task.h"
|
#include "tasks/Task.h"
|
||||||
|
|
||||||
class InstanceCopyTask : public InstanceTask {
|
class InstanceCopyTask : public InstanceTask
|
||||||
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
explicit InstanceCopyTask(InstancePtr origInstance, const InstanceCopyPrefs& prefs);
|
explicit InstanceCopyTask(InstancePtr origInstance, const InstanceCopyPrefs& prefs);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
//! Entry point for tasks.
|
//! Entry point for tasks.
|
||||||
virtual void executeTask() override;
|
virtual void executeTask() override;
|
||||||
void copyFinished();
|
void copyFinished();
|
||||||
void copyAborted();
|
void copyAborted();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
/* data */
|
/* data */
|
||||||
InstancePtr m_origInstance;
|
InstancePtr m_origInstance;
|
||||||
QFuture<bool> m_copyFuture;
|
QFuture<bool> m_copyFuture;
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
* Copyright (c) 2022 flowln <flowlnlnln@gmail.com>
|
* Copyright (c) 2022 flowln <flowlnlnln@gmail.com>
|
||||||
*
|
*
|
||||||
@ -45,14 +45,11 @@
|
|||||||
#include "icons/IconList.h"
|
#include "icons/IconList.h"
|
||||||
#include "icons/IconUtils.h"
|
#include "icons/IconUtils.h"
|
||||||
|
|
||||||
#include "modplatform/flame/FlameInstanceCreationTask.h"
|
|
||||||
#include "modplatform/modrinth/ModrinthInstanceCreationTask.h"
|
|
||||||
#include "modplatform/technic/TechnicPackProcessor.h"
|
#include "modplatform/technic/TechnicPackProcessor.h"
|
||||||
|
#include "modplatform/modrinth/ModrinthInstanceCreationTask.h"
|
||||||
|
#include "modplatform/flame/FlameInstanceCreationTask.h"
|
||||||
|
|
||||||
#include "settings/INISettingsObject.h"
|
#include "settings/INISettingsObject.h"
|
||||||
#include "tasks/Task.h"
|
|
||||||
|
|
||||||
#include "net/ApiDownload.h"
|
|
||||||
|
|
||||||
#include <QtConcurrentRun>
|
#include <QtConcurrentRun>
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
@ -91,27 +88,25 @@ void InstanceImportTask::executeTask()
|
|||||||
setStatus(tr("Downloading modpack:\n%1").arg(m_sourceUrl.toString()));
|
setStatus(tr("Downloading modpack:\n%1").arg(m_sourceUrl.toString()));
|
||||||
m_downloadRequired = true;
|
m_downloadRequired = true;
|
||||||
|
|
||||||
downloadFromUrl();
|
const QString path(m_sourceUrl.host() + '/' + m_sourceUrl.path());
|
||||||
|
|
||||||
|
auto entry = APPLICATION->metacache()->resolveEntry("general", path);
|
||||||
|
entry->setStale(true);
|
||||||
|
m_archivePath = entry->getFullPath();
|
||||||
|
|
||||||
|
m_filesNetJob.reset(new NetJob(tr("Modpack download"), APPLICATION->network()));
|
||||||
|
m_filesNetJob->addNetAction(Net::Download::makeCached(m_sourceUrl, entry));
|
||||||
|
|
||||||
|
connect(m_filesNetJob.get(), &NetJob::succeeded, this, &InstanceImportTask::downloadSucceeded);
|
||||||
|
connect(m_filesNetJob.get(), &NetJob::progress, this, &InstanceImportTask::downloadProgressChanged);
|
||||||
|
connect(m_filesNetJob.get(), &NetJob::stepProgress, this, &InstanceImportTask::propogateStepProgress);
|
||||||
|
connect(m_filesNetJob.get(), &NetJob::failed, this, &InstanceImportTask::downloadFailed);
|
||||||
|
connect(m_filesNetJob.get(), &NetJob::aborted, this, &InstanceImportTask::downloadAborted);
|
||||||
|
|
||||||
|
m_filesNetJob->start();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstanceImportTask::downloadFromUrl()
|
|
||||||
{
|
|
||||||
const QString path = m_sourceUrl.host() + '/' + m_sourceUrl.path();
|
|
||||||
auto entry = APPLICATION->metacache()->resolveEntry("general", path);
|
|
||||||
entry->setStale(true);
|
|
||||||
m_filesNetJob.reset(new NetJob(tr("Modpack download"), APPLICATION->network()));
|
|
||||||
m_filesNetJob->addNetAction(Net::ApiDownload::makeCached(m_sourceUrl, entry));
|
|
||||||
m_archivePath = entry->getFullPath();
|
|
||||||
|
|
||||||
connect(m_filesNetJob.get(), &NetJob::succeeded, this, &InstanceImportTask::downloadSucceeded);
|
|
||||||
connect(m_filesNetJob.get(), &NetJob::progress, this, &InstanceImportTask::downloadProgressChanged);
|
|
||||||
connect(m_filesNetJob.get(), &NetJob::stepProgress, this, &InstanceImportTask::propagateStepProgress);
|
|
||||||
connect(m_filesNetJob.get(), &NetJob::failed, this, &InstanceImportTask::downloadFailed);
|
|
||||||
connect(m_filesNetJob.get(), &NetJob::aborted, this, &InstanceImportTask::downloadAborted);
|
|
||||||
m_filesNetJob->start();
|
|
||||||
}
|
|
||||||
|
|
||||||
void InstanceImportTask::downloadSucceeded()
|
void InstanceImportTask::downloadSucceeded()
|
||||||
{
|
{
|
||||||
processZipPack();
|
processZipPack();
|
||||||
@ -143,7 +138,8 @@ void InstanceImportTask::processZipPack()
|
|||||||
|
|
||||||
// open the zip and find relevant files in it
|
// open the zip and find relevant files in it
|
||||||
m_packZip.reset(new QuaZip(m_archivePath));
|
m_packZip.reset(new QuaZip(m_archivePath));
|
||||||
if (!m_packZip->open(QuaZip::mdUnzip)) {
|
if (!m_packZip->open(QuaZip::mdUnzip))
|
||||||
|
{
|
||||||
emitFailed(tr("Unable to open supplied modpack zip file."));
|
emitFailed(tr("Unable to open supplied modpack zip file."));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -157,40 +153,44 @@ void InstanceImportTask::processZipPack()
|
|||||||
|
|
||||||
// NOTE: Prioritize modpack platforms that aren't searched for recursively.
|
// NOTE: Prioritize modpack platforms that aren't searched for recursively.
|
||||||
// Especially Flame has a very common filename for its manifest, which may appear inside overrides for example
|
// Especially Flame has a very common filename for its manifest, which may appear inside overrides for example
|
||||||
if (modrinthFound) {
|
if(modrinthFound)
|
||||||
|
{
|
||||||
// process as Modrinth pack
|
// process as Modrinth pack
|
||||||
qDebug() << "Modrinth:" << modrinthFound;
|
qDebug() << "Modrinth:" << modrinthFound;
|
||||||
m_modpackType = ModpackType::Modrinth;
|
m_modpackType = ModpackType::Modrinth;
|
||||||
} else if (technicFound) {
|
}
|
||||||
|
else if (technicFound)
|
||||||
|
{
|
||||||
// process as Technic pack
|
// process as Technic pack
|
||||||
qDebug() << "Technic:" << technicFound;
|
qDebug() << "Technic:" << technicFound;
|
||||||
extractDir.mkpath(".minecraft");
|
extractDir.mkpath(".minecraft");
|
||||||
extractDir.cd(".minecraft");
|
extractDir.cd(".minecraft");
|
||||||
m_modpackType = ModpackType::Technic;
|
m_modpackType = ModpackType::Technic;
|
||||||
} else {
|
}
|
||||||
QStringList paths_to_ignore{ "overrides/" };
|
else
|
||||||
|
{
|
||||||
|
QStringList paths_to_ignore { "overrides/" };
|
||||||
|
|
||||||
if (QString mmcRoot = MMCZip::findFolderOfFileInZip(m_packZip.get(), "instance.cfg", paths_to_ignore); !mmcRoot.isNull()) {
|
if (QString mmcRoot = MMCZip::findFolderOfFileInZip(m_packZip.get(), "instance.cfg", paths_to_ignore); !mmcRoot.isNull()) {
|
||||||
// process as MultiMC instance/pack
|
// process as MultiMC instance/pack
|
||||||
qDebug() << "MultiMC:" << mmcRoot;
|
qDebug() << "MultiMC:" << mmcRoot;
|
||||||
root = mmcRoot;
|
root = mmcRoot;
|
||||||
m_modpackType = ModpackType::MultiMC;
|
m_modpackType = ModpackType::MultiMC;
|
||||||
} else if (QString flameRoot = MMCZip::findFolderOfFileInZip(m_packZip.get(), "manifest.json", paths_to_ignore);
|
} else if (QString flameRoot = MMCZip::findFolderOfFileInZip(m_packZip.get(), "manifest.json", paths_to_ignore); !flameRoot.isNull()) {
|
||||||
!flameRoot.isNull()) {
|
|
||||||
// process as Flame pack
|
// process as Flame pack
|
||||||
qDebug() << "Flame:" << flameRoot;
|
qDebug() << "Flame:" << flameRoot;
|
||||||
root = flameRoot;
|
root = flameRoot;
|
||||||
m_modpackType = ModpackType::Flame;
|
m_modpackType = ModpackType::Flame;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (m_modpackType == ModpackType::Unknown) {
|
if(m_modpackType == ModpackType::Unknown)
|
||||||
|
{
|
||||||
emitFailed(tr("Archive does not contain a recognized modpack type."));
|
emitFailed(tr("Archive does not contain a recognized modpack type."));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// make sure we extract just the pack
|
// make sure we extract just the pack
|
||||||
m_extractFuture =
|
m_extractFuture = QtConcurrent::run(QThreadPool::globalInstance(), MMCZip::extractSubDir, m_packZip.get(), root, extractDir.absolutePath());
|
||||||
QtConcurrent::run(QThreadPool::globalInstance(), MMCZip::extractSubDir, m_packZip.get(), root, extractDir.absolutePath());
|
|
||||||
connect(&m_extractFutureWatcher, &QFutureWatcher<QStringList>::finished, this, &InstanceImportTask::extractFinished);
|
connect(&m_extractFutureWatcher, &QFutureWatcher<QStringList>::finished, this, &InstanceImportTask::extractFinished);
|
||||||
m_extractFutureWatcher.setFuture(m_extractFuture);
|
m_extractFutureWatcher.setFuture(m_extractFuture);
|
||||||
}
|
}
|
||||||
@ -210,28 +210,37 @@ void InstanceImportTask::extractFinished()
|
|||||||
|
|
||||||
qDebug() << "Fixing permissions for extracted pack files...";
|
qDebug() << "Fixing permissions for extracted pack files...";
|
||||||
QDirIterator it(extractDir, QDirIterator::Subdirectories);
|
QDirIterator it(extractDir, QDirIterator::Subdirectories);
|
||||||
while (it.hasNext()) {
|
while (it.hasNext())
|
||||||
|
{
|
||||||
auto filepath = it.next();
|
auto filepath = it.next();
|
||||||
QFileInfo file(filepath);
|
QFileInfo file(filepath);
|
||||||
auto permissions = QFile::permissions(filepath);
|
auto permissions = QFile::permissions(filepath);
|
||||||
auto origPermissions = permissions;
|
auto origPermissions = permissions;
|
||||||
if (file.isDir()) {
|
if(file.isDir())
|
||||||
|
{
|
||||||
// Folder +rwx for current user
|
// Folder +rwx for current user
|
||||||
permissions |= QFileDevice::Permission::ReadUser | QFileDevice::Permission::WriteUser | QFileDevice::Permission::ExeUser;
|
permissions |= QFileDevice::Permission::ReadUser | QFileDevice::Permission::WriteUser | QFileDevice::Permission::ExeUser;
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
// File +rw for current user
|
// File +rw for current user
|
||||||
permissions |= QFileDevice::Permission::ReadUser | QFileDevice::Permission::WriteUser;
|
permissions |= QFileDevice::Permission::ReadUser | QFileDevice::Permission::WriteUser;
|
||||||
}
|
}
|
||||||
if (origPermissions != permissions) {
|
if(origPermissions != permissions)
|
||||||
if (!QFile::setPermissions(filepath, permissions)) {
|
{
|
||||||
|
if(!QFile::setPermissions(filepath, permissions))
|
||||||
|
{
|
||||||
logWarning(tr("Could not fix permissions for %1").arg(filepath));
|
logWarning(tr("Could not fix permissions for %1").arg(filepath));
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
qDebug() << "Fixed" << filepath;
|
qDebug() << "Fixed" << filepath;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (m_modpackType) {
|
switch(m_modpackType)
|
||||||
|
{
|
||||||
case ModpackType::MultiMC:
|
case ModpackType::MultiMC:
|
||||||
processMultiMC();
|
processMultiMC();
|
||||||
return;
|
return;
|
||||||
@ -267,8 +276,7 @@ void InstanceImportTask::processFlame()
|
|||||||
if (original_instance_id_it != m_extra_info.constEnd())
|
if (original_instance_id_it != m_extra_info.constEnd())
|
||||||
original_instance_id = original_instance_id_it.value();
|
original_instance_id = original_instance_id_it.value();
|
||||||
|
|
||||||
inst_creation_task =
|
inst_creation_task = makeShared<FlameCreationTask>(m_stagingPath, m_globalSettings, m_parent, pack_id, pack_version_id, original_instance_id);
|
||||||
makeShared<FlameCreationTask>(m_stagingPath, m_globalSettings, m_parent, pack_id, pack_version_id, original_instance_id);
|
|
||||||
} else {
|
} else {
|
||||||
// FIXME: Find a way to get IDs in directly imported ZIPs
|
// FIXME: Find a way to get IDs in directly imported ZIPs
|
||||||
inst_creation_task = makeShared<FlameCreationTask>(m_stagingPath, m_globalSettings, m_parent, QString(), QString());
|
inst_creation_task = makeShared<FlameCreationTask>(m_stagingPath, m_globalSettings, m_parent, QString(), QString());
|
||||||
@ -278,14 +286,14 @@ void InstanceImportTask::processFlame()
|
|||||||
inst_creation_task->setIcon(m_instIcon);
|
inst_creation_task->setIcon(m_instIcon);
|
||||||
inst_creation_task->setGroup(m_instGroup);
|
inst_creation_task->setGroup(m_instGroup);
|
||||||
inst_creation_task->setConfirmUpdate(shouldConfirmUpdate());
|
inst_creation_task->setConfirmUpdate(shouldConfirmUpdate());
|
||||||
|
|
||||||
connect(inst_creation_task.get(), &Task::succeeded, this, [this, inst_creation_task] {
|
connect(inst_creation_task.get(), &Task::succeeded, this, [this, inst_creation_task] {
|
||||||
setOverride(inst_creation_task->shouldOverride(), inst_creation_task->originalInstanceID());
|
setOverride(inst_creation_task->shouldOverride(), inst_creation_task->originalInstanceID());
|
||||||
emitSucceeded();
|
emitSucceeded();
|
||||||
});
|
});
|
||||||
connect(inst_creation_task.get(), &Task::failed, this, &InstanceImportTask::emitFailed);
|
connect(inst_creation_task.get(), &Task::failed, this, &InstanceImportTask::emitFailed);
|
||||||
connect(inst_creation_task.get(), &Task::progress, this, &InstanceImportTask::setProgress);
|
connect(inst_creation_task.get(), &Task::progress, this, &InstanceImportTask::setProgress);
|
||||||
connect(inst_creation_task.get(), &Task::stepProgress, this, &InstanceImportTask::propagateStepProgress);
|
connect(inst_creation_task.get(), &Task::stepProgress, this, &InstanceImportTask::propogateStepProgress);
|
||||||
connect(inst_creation_task.get(), &Task::status, this, &InstanceImportTask::setStatus);
|
connect(inst_creation_task.get(), &Task::status, this, &InstanceImportTask::setStatus);
|
||||||
connect(inst_creation_task.get(), &Task::details, this, &InstanceImportTask::setDetails);
|
connect(inst_creation_task.get(), &Task::details, this, &InstanceImportTask::setDetails);
|
||||||
|
|
||||||
@ -354,8 +362,7 @@ void InstanceImportTask::processModrinth()
|
|||||||
if (original_instance_id_it != m_extra_info.constEnd())
|
if (original_instance_id_it != m_extra_info.constEnd())
|
||||||
original_instance_id = original_instance_id_it.value();
|
original_instance_id = original_instance_id_it.value();
|
||||||
|
|
||||||
inst_creation_task =
|
inst_creation_task = new ModrinthCreationTask(m_stagingPath, m_globalSettings, m_parent, pack_id, pack_version_id, original_instance_id);
|
||||||
new ModrinthCreationTask(m_stagingPath, m_globalSettings, m_parent, pack_id, pack_version_id, original_instance_id);
|
|
||||||
} else {
|
} else {
|
||||||
QString pack_id;
|
QString pack_id;
|
||||||
if (!m_sourceUrl.isEmpty()) {
|
if (!m_sourceUrl.isEmpty()) {
|
||||||
@ -371,14 +378,14 @@ void InstanceImportTask::processModrinth()
|
|||||||
inst_creation_task->setIcon(m_instIcon);
|
inst_creation_task->setIcon(m_instIcon);
|
||||||
inst_creation_task->setGroup(m_instGroup);
|
inst_creation_task->setGroup(m_instGroup);
|
||||||
inst_creation_task->setConfirmUpdate(shouldConfirmUpdate());
|
inst_creation_task->setConfirmUpdate(shouldConfirmUpdate());
|
||||||
|
|
||||||
connect(inst_creation_task, &Task::succeeded, this, [this, inst_creation_task] {
|
connect(inst_creation_task, &Task::succeeded, this, [this, inst_creation_task] {
|
||||||
setOverride(inst_creation_task->shouldOverride(), inst_creation_task->originalInstanceID());
|
setOverride(inst_creation_task->shouldOverride(), inst_creation_task->originalInstanceID());
|
||||||
emitSucceeded();
|
emitSucceeded();
|
||||||
});
|
});
|
||||||
connect(inst_creation_task, &Task::failed, this, &InstanceImportTask::emitFailed);
|
connect(inst_creation_task, &Task::failed, this, &InstanceImportTask::emitFailed);
|
||||||
connect(inst_creation_task, &Task::progress, this, &InstanceImportTask::setProgress);
|
connect(inst_creation_task, &Task::progress, this, &InstanceImportTask::setProgress);
|
||||||
connect(inst_creation_task, &Task::stepProgress, this, &InstanceImportTask::propagateStepProgress);
|
connect(inst_creation_task, &Task::stepProgress, this, &InstanceImportTask::propogateStepProgress);
|
||||||
connect(inst_creation_task, &Task::status, this, &InstanceImportTask::setStatus);
|
connect(inst_creation_task, &Task::status, this, &InstanceImportTask::setStatus);
|
||||||
connect(inst_creation_task, &Task::details, this, &InstanceImportTask::setDetails);
|
connect(inst_creation_task, &Task::details, this, &InstanceImportTask::setDetails);
|
||||||
connect(inst_creation_task, &Task::finished, inst_creation_task, &InstanceCreationTask::deleteLater);
|
connect(inst_creation_task, &Task::finished, inst_creation_task, &InstanceCreationTask::deleteLater);
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
@ -35,49 +35,54 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include "InstanceTask.h"
|
||||||
|
#include "net/NetJob.h"
|
||||||
|
#include <QUrl>
|
||||||
#include <QFuture>
|
#include <QFuture>
|
||||||
#include <QFutureWatcher>
|
#include <QFutureWatcher>
|
||||||
#include <QUrl>
|
#include "settings/SettingsObject.h"
|
||||||
#include "InstanceTask.h"
|
|
||||||
#include "QObjectPtr.h"
|
#include "QObjectPtr.h"
|
||||||
#include "modplatform/flame/PackManifest.h"
|
#include "modplatform/flame/PackManifest.h"
|
||||||
#include "net/NetJob.h"
|
|
||||||
#include "settings/SettingsObject.h"
|
|
||||||
|
|
||||||
#include <optional>
|
#include <optional>
|
||||||
|
|
||||||
class QuaZip;
|
class QuaZip;
|
||||||
namespace Flame {
|
namespace Flame
|
||||||
class FileResolvingTask;
|
{
|
||||||
|
class FileResolvingTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
class InstanceImportTask : public InstanceTask {
|
class InstanceImportTask : public InstanceTask
|
||||||
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
explicit InstanceImportTask(const QUrl sourceUrl, QWidget* parent = nullptr, QMap<QString, QString>&& extra_info = {});
|
explicit InstanceImportTask(const QUrl sourceUrl, QWidget* parent = nullptr, QMap<QString, QString>&& extra_info = {});
|
||||||
|
|
||||||
bool abort() override;
|
bool abort() override;
|
||||||
const QVector<Flame::File>& getBlockedFiles() const { return m_blockedMods; }
|
const QVector<Flame::File> &getBlockedFiles() const
|
||||||
|
{
|
||||||
|
return m_blockedMods;
|
||||||
|
}
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
//! Entry point for tasks.
|
//! Entry point for tasks.
|
||||||
virtual void executeTask() override;
|
virtual void executeTask() override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void processZipPack();
|
void processZipPack();
|
||||||
void processMultiMC();
|
void processMultiMC();
|
||||||
void processTechnic();
|
void processTechnic();
|
||||||
void processFlame();
|
void processFlame();
|
||||||
void processModrinth();
|
void processModrinth();
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void downloadSucceeded();
|
void downloadSucceeded();
|
||||||
void downloadFailed(QString reason);
|
void downloadFailed(QString reason);
|
||||||
void downloadProgressChanged(qint64 current, qint64 total);
|
void downloadProgressChanged(qint64 current, qint64 total);
|
||||||
void downloadAborted();
|
void downloadAborted();
|
||||||
void extractFinished();
|
void extractFinished();
|
||||||
|
|
||||||
private: /* data */
|
private: /* data */
|
||||||
NetJob::Ptr m_filesNetJob;
|
NetJob::Ptr m_filesNetJob;
|
||||||
shared_qobject_ptr<Flame::FileResolvingTask> m_modIdResolver;
|
shared_qobject_ptr<Flame::FileResolvingTask> m_modIdResolver;
|
||||||
QUrl m_sourceUrl;
|
QUrl m_sourceUrl;
|
||||||
@ -87,7 +92,7 @@ class InstanceImportTask : public InstanceTask {
|
|||||||
QFuture<std::optional<QStringList>> m_extractFuture;
|
QFuture<std::optional<QStringList>> m_extractFuture;
|
||||||
QFutureWatcher<std::optional<QStringList>> m_extractFutureWatcher;
|
QFutureWatcher<std::optional<QStringList>> m_extractFutureWatcher;
|
||||||
QVector<Flame::File> m_blockedMods;
|
QVector<Flame::File> m_blockedMods;
|
||||||
enum class ModpackType {
|
enum class ModpackType{
|
||||||
Unknown,
|
Unknown,
|
||||||
MultiMC,
|
MultiMC,
|
||||||
Technic,
|
Technic,
|
||||||
@ -99,7 +104,6 @@ class InstanceImportTask : public InstanceTask {
|
|||||||
// the source URL / the resource it points to alone.
|
// the source URL / the resource it points to alone.
|
||||||
QMap<QString, QString> m_extra_info;
|
QMap<QString, QString> m_extra_info;
|
||||||
|
|
||||||
// FIXME: nuke
|
//FIXME: nuke
|
||||||
QWidget* m_parent;
|
QWidget* m_parent;
|
||||||
void downloadFromUrl();
|
|
||||||
};
|
};
|
||||||
|
@ -1,8 +1,7 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
* Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me>
|
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -42,9 +41,9 @@
|
|||||||
#include <QJsonArray>
|
#include <QJsonArray>
|
||||||
#include <QJsonDocument>
|
#include <QJsonDocument>
|
||||||
#include <QMimeData>
|
#include <QMimeData>
|
||||||
#include <QPair>
|
|
||||||
#include <QSet>
|
#include <QSet>
|
||||||
#include <QStack>
|
#include <QStack>
|
||||||
|
#include <QPair>
|
||||||
#include <QTextStream>
|
#include <QTextStream>
|
||||||
#include <QThread>
|
#include <QThread>
|
||||||
#include <QTimer>
|
#include <QTimer>
|
||||||
@ -97,11 +96,7 @@ Qt::DropActions InstanceList::supportedDropActions() const
|
|||||||
return Qt::MoveAction;
|
return Qt::MoveAction;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool InstanceList::canDropMimeData(const QMimeData* data,
|
bool InstanceList::canDropMimeData(const QMimeData* data, Qt::DropAction action, int row, int column, const QModelIndex& parent) const
|
||||||
[[maybe_unused]] Qt::DropAction action,
|
|
||||||
[[maybe_unused]] int row,
|
|
||||||
[[maybe_unused]] int column,
|
|
||||||
[[maybe_unused]] const QModelIndex& parent) const
|
|
||||||
{
|
{
|
||||||
if (data && data->hasFormat("application/x-instanceid")) {
|
if (data && data->hasFormat("application/x-instanceid")) {
|
||||||
return true;
|
return true;
|
||||||
@ -109,11 +104,7 @@ bool InstanceList::canDropMimeData(const QMimeData* data,
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool InstanceList::dropMimeData(const QMimeData* data,
|
bool InstanceList::dropMimeData(const QMimeData* data, Qt::DropAction action, int row, int column, const QModelIndex& parent)
|
||||||
[[maybe_unused]] Qt::DropAction action,
|
|
||||||
[[maybe_unused]] int row,
|
|
||||||
[[maybe_unused]] int column,
|
|
||||||
[[maybe_unused]] const QModelIndex& parent)
|
|
||||||
{
|
{
|
||||||
if (data && data->hasFormat("application/x-instanceid")) {
|
if (data && data->hasFormat("application/x-instanceid")) {
|
||||||
return true;
|
return true;
|
||||||
@ -138,7 +129,7 @@ QMimeData* InstanceList::mimeData(const QModelIndexList& indexes) const
|
|||||||
return mimeData;
|
return mimeData;
|
||||||
}
|
}
|
||||||
|
|
||||||
QStringList InstanceList::getLinkedInstancesById(const QString& id) const
|
QStringList InstanceList::getLinkedInstancesById(const QString &id) const
|
||||||
{
|
{
|
||||||
QStringList linkedInstances;
|
QStringList linkedInstances;
|
||||||
for (auto inst : m_instances) {
|
for (auto inst : m_instances) {
|
||||||
@ -167,34 +158,42 @@ QVariant InstanceList::data(const QModelIndex& index, int role) const
|
|||||||
if (!index.isValid()) {
|
if (!index.isValid()) {
|
||||||
return QVariant();
|
return QVariant();
|
||||||
}
|
}
|
||||||
BaseInstance* pdata = static_cast<BaseInstance*>(index.internalPointer());
|
BaseInstance *pdata = static_cast<BaseInstance *>(index.internalPointer());
|
||||||
switch (role) {
|
switch (role)
|
||||||
case InstancePointerRole: {
|
{
|
||||||
QVariant v = QVariant::fromValue((void*)pdata);
|
case InstancePointerRole:
|
||||||
return v;
|
{
|
||||||
}
|
QVariant v = QVariant::fromValue((void *)pdata);
|
||||||
case InstanceIDRole: {
|
return v;
|
||||||
return pdata->id();
|
}
|
||||||
}
|
case InstanceIDRole:
|
||||||
case Qt::EditRole:
|
{
|
||||||
case Qt::DisplayRole: {
|
return pdata->id();
|
||||||
return pdata->name();
|
}
|
||||||
}
|
case Qt::EditRole:
|
||||||
case Qt::AccessibleTextRole: {
|
case Qt::DisplayRole:
|
||||||
return tr("%1 Instance").arg(pdata->name());
|
{
|
||||||
}
|
return pdata->name();
|
||||||
case Qt::ToolTipRole: {
|
}
|
||||||
return pdata->instanceRoot();
|
case Qt::AccessibleTextRole:
|
||||||
}
|
{
|
||||||
case Qt::DecorationRole: {
|
return tr("%1 Instance").arg(pdata->name());
|
||||||
return pdata->iconKey();
|
}
|
||||||
}
|
case Qt::ToolTipRole:
|
||||||
// HACK: see InstanceView.h in gui!
|
{
|
||||||
case GroupRole: {
|
return pdata->instanceRoot();
|
||||||
return getInstanceGroup(pdata->id());
|
}
|
||||||
}
|
case Qt::DecorationRole:
|
||||||
default:
|
{
|
||||||
break;
|
return pdata->iconKey();
|
||||||
|
}
|
||||||
|
// HACK: see InstanceView.h in gui!
|
||||||
|
case GroupRole:
|
||||||
|
{
|
||||||
|
return getInstanceGroup(pdata->id());
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
return QVariant();
|
return QVariant();
|
||||||
}
|
}
|
||||||
@ -238,11 +237,8 @@ GroupId InstanceList::getInstanceGroup(const InstanceId& id) const
|
|||||||
return GroupId();
|
return GroupId();
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstanceList::setInstanceGroup(const InstanceId& id, GroupId name)
|
void InstanceList::setInstanceGroup(const InstanceId& id, const GroupId& name)
|
||||||
{
|
{
|
||||||
if (name.isEmpty() && !name.isNull())
|
|
||||||
name = QString();
|
|
||||||
|
|
||||||
auto inst = getInstanceById(id);
|
auto inst = getInstanceById(id);
|
||||||
if (!inst) {
|
if (!inst) {
|
||||||
qDebug() << "Attempt to set a null instance's group";
|
qDebug() << "Attempt to set a null instance's group";
|
||||||
@ -253,7 +249,6 @@ void InstanceList::setInstanceGroup(const InstanceId& id, GroupId name)
|
|||||||
auto iter = m_instanceGroupIndex.find(inst->id());
|
auto iter = m_instanceGroupIndex.find(inst->id());
|
||||||
if (iter != m_instanceGroupIndex.end()) {
|
if (iter != m_instanceGroupIndex.end()) {
|
||||||
if (*iter != name) {
|
if (*iter != name) {
|
||||||
decreaseGroupCount(*iter);
|
|
||||||
*iter = name;
|
*iter = name;
|
||||||
changed = true;
|
changed = true;
|
||||||
}
|
}
|
||||||
@ -263,7 +258,7 @@ void InstanceList::setInstanceGroup(const InstanceId& id, GroupId name)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (changed) {
|
if (changed) {
|
||||||
increaseGroupCount(name);
|
m_groupNameCache.insert(name);
|
||||||
auto idx = getInstIndex(inst.get());
|
auto idx = getInstIndex(inst.get());
|
||||||
emit dataChanged(index(idx), index(idx), { GroupRole });
|
emit dataChanged(index(idx), index(idx), { GroupRole });
|
||||||
saveGroupList();
|
saveGroupList();
|
||||||
@ -272,55 +267,29 @@ void InstanceList::setInstanceGroup(const InstanceId& id, GroupId name)
|
|||||||
|
|
||||||
QStringList InstanceList::getGroups()
|
QStringList InstanceList::getGroups()
|
||||||
{
|
{
|
||||||
return m_groupNameCache.keys();
|
return m_groupNameCache.values();
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstanceList::deleteGroup(const GroupId& name)
|
void InstanceList::deleteGroup(const QString& name)
|
||||||
{
|
{
|
||||||
m_groupNameCache.remove(name);
|
|
||||||
m_collapsedGroups.remove(name);
|
|
||||||
|
|
||||||
bool removed = false;
|
bool removed = false;
|
||||||
qDebug() << "Delete group" << name;
|
qDebug() << "Delete group" << name;
|
||||||
for (auto& instance : m_instances) {
|
for (auto& instance : m_instances) {
|
||||||
const QString& instID = instance->id();
|
const auto& instID = instance->id();
|
||||||
const QString instGroupName = getInstanceGroup(instID);
|
auto instGroupName = getInstanceGroup(instID);
|
||||||
if (instGroupName == name) {
|
if (instGroupName == name) {
|
||||||
m_instanceGroupIndex.remove(instID);
|
m_instanceGroupIndex.remove(instID);
|
||||||
qDebug() << "Remove" << instID << "from group" << name;
|
qDebug() << "Remove" << instID << "from group" << name;
|
||||||
removed = true;
|
removed = true;
|
||||||
auto idx = getInstIndex(instance.get());
|
auto idx = getInstIndex(instance.get());
|
||||||
if (idx >= 0)
|
if (idx > 0) {
|
||||||
emit dataChanged(index(idx), index(idx), { GroupRole });
|
emit dataChanged(index(idx), index(idx), { GroupRole });
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (removed)
|
if (removed) {
|
||||||
saveGroupList();
|
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)
|
bool InstanceList::isGroupCollapsed(const QString& group)
|
||||||
@ -336,13 +305,12 @@ bool InstanceList::trashInstance(const InstanceId& id)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
QString cachedGroupId = m_instanceGroupIndex[id];
|
auto cachedGroupId = m_instanceGroupIndex[id];
|
||||||
|
|
||||||
qDebug() << "Will trash instance" << id;
|
qDebug() << "Will trash instance" << id;
|
||||||
QString trashedLoc;
|
QString trashedLoc;
|
||||||
|
|
||||||
if (m_instanceGroupIndex.remove(id)) {
|
if (m_instanceGroupIndex.remove(id)) {
|
||||||
decreaseGroupCount(cachedGroupId);
|
|
||||||
saveGroupList();
|
saveGroupList();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -352,18 +320,16 @@ bool InstanceList::trashInstance(const InstanceId& id)
|
|||||||
}
|
}
|
||||||
|
|
||||||
qDebug() << "Instance" << id << "has been trashed by the launcher.";
|
qDebug() << "Instance" << id << "has been trashed by the launcher.";
|
||||||
m_trashHistory.push({ id, inst->instanceRoot(), trashedLoc, cachedGroupId });
|
m_trashHistory.push({id, inst->instanceRoot(), trashedLoc, cachedGroupId});
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool InstanceList::trashedSomething()
|
bool InstanceList::trashedSomething() {
|
||||||
{
|
|
||||||
return !m_trashHistory.empty();
|
return !m_trashHistory.empty();
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstanceList::undoTrashInstance()
|
void InstanceList::undoTrashInstance() {
|
||||||
{
|
|
||||||
if (m_trashHistory.empty()) {
|
if (m_trashHistory.empty()) {
|
||||||
qWarning() << "Nothing to recover from trash.";
|
qWarning() << "Nothing to recover from trash.";
|
||||||
return;
|
return;
|
||||||
@ -380,7 +346,7 @@ void InstanceList::undoTrashInstance()
|
|||||||
QFile(top.trashPath).rename(top.polyPath);
|
QFile(top.trashPath).rename(top.polyPath);
|
||||||
|
|
||||||
m_instanceGroupIndex[top.id] = top.groupName;
|
m_instanceGroupIndex[top.id] = top.groupName;
|
||||||
increaseGroupCount(top.groupName);
|
m_groupNameCache.insert(top.groupName);
|
||||||
|
|
||||||
saveGroupList();
|
saveGroupList();
|
||||||
emit instancesChanged();
|
emit instancesChanged();
|
||||||
@ -394,10 +360,7 @@ void InstanceList::deleteInstance(const InstanceId& id)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
QString cachedGroupId = m_instanceGroupIndex[id];
|
|
||||||
|
|
||||||
if (m_instanceGroupIndex.remove(id)) {
|
if (m_instanceGroupIndex.remove(id)) {
|
||||||
decreaseGroupCount(cachedGroupId);
|
|
||||||
saveGroupList();
|
saveGroupList();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -595,7 +558,7 @@ InstancePtr InstanceList::getInstanceByManagedName(const QString& managed_name)
|
|||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
QModelIndex InstanceList::getInstanceIndexById(const QString& id) const
|
QModelIndex InstanceList::getInstanceIndexById(const QString &id) const
|
||||||
{
|
{
|
||||||
return index(getInstIndex(getInstanceById(id).get()));
|
return index(getInstIndex(getInstanceById(id).get()));
|
||||||
}
|
}
|
||||||
@ -634,36 +597,19 @@ InstancePtr InstanceList::loadInstance(const InstanceId& id)
|
|||||||
|
|
||||||
QString inst_type = instanceSettings->get("InstanceType").toString();
|
QString inst_type = instanceSettings->get("InstanceType").toString();
|
||||||
|
|
||||||
// NOTE: Some PolyMC versions didn't save the InstanceType properly. We will just bank on the probability that this is probably a OneSix
|
// NOTE: Some PolyMC versions didn't save the InstanceType properly. We will just bank on the probability that this is probably a OneSix instance
|
||||||
// instance
|
if (inst_type == "OneSix" || inst_type.isEmpty())
|
||||||
if (inst_type == "OneSix" || inst_type.isEmpty()) {
|
{
|
||||||
inst.reset(new MinecraftInstance(m_globalSettings, instanceSettings, instanceRoot));
|
inst.reset(new MinecraftInstance(m_globalSettings, instanceSettings, instanceRoot));
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
inst.reset(new NullInstance(m_globalSettings, instanceSettings, instanceRoot));
|
inst.reset(new NullInstance(m_globalSettings, instanceSettings, instanceRoot));
|
||||||
}
|
}
|
||||||
qDebug() << "Loaded instance " << inst->name() << " from " << inst->instanceRoot();
|
qDebug() << "Loaded instance " << inst->name() << " from " << inst->instanceRoot();
|
||||||
return inst;
|
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()
|
void InstanceList::saveGroupList()
|
||||||
{
|
{
|
||||||
qDebug() << "Will save group list now.";
|
qDebug() << "Will save group list now.";
|
||||||
@ -675,7 +621,7 @@ void InstanceList::saveGroupList()
|
|||||||
QString groupFileName = m_instDir + "/instgroups.json";
|
QString groupFileName = m_instDir + "/instgroups.json";
|
||||||
QMap<QString, QSet<QString>> reverseGroupMap;
|
QMap<QString, QSet<QString>> reverseGroupMap;
|
||||||
for (auto iter = m_instanceGroupIndex.begin(); iter != m_instanceGroupIndex.end(); iter++) {
|
for (auto iter = m_instanceGroupIndex.begin(); iter != m_instanceGroupIndex.end(); iter++) {
|
||||||
const QString& id = iter.key();
|
QString id = iter.key();
|
||||||
QString group = iter.value();
|
QString group = iter.value();
|
||||||
if (group.isEmpty())
|
if (group.isEmpty())
|
||||||
continue;
|
continue;
|
||||||
@ -765,22 +711,17 @@ void InstanceList::loadGroupList()
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QSet<QString> groupSet;
|
||||||
m_instanceGroupIndex.clear();
|
m_instanceGroupIndex.clear();
|
||||||
m_groupNameCache.clear();
|
|
||||||
|
|
||||||
// Iterate through all the groups.
|
// Iterate through all the groups.
|
||||||
QJsonObject groupMapping = rootObj.value("groups").toObject();
|
QJsonObject groupMapping = rootObj.value("groups").toObject();
|
||||||
for (QJsonObject::iterator iter = groupMapping.begin(); iter != groupMapping.end(); iter++) {
|
for (QJsonObject::iterator iter = groupMapping.begin(); iter != groupMapping.end(); iter++) {
|
||||||
QString groupName = iter.key();
|
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 not an object, complain and skip to the next one.
|
||||||
if (!iter.value().isObject()) {
|
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;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -792,19 +733,23 @@ void InstanceList::loadGroupList()
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// keep a list/set of groups for choosing
|
||||||
|
groupSet.insert(groupName);
|
||||||
|
|
||||||
auto hidden = groupObj.value("hidden").toBool(false);
|
auto hidden = groupObj.value("hidden").toBool(false);
|
||||||
if (hidden)
|
if (hidden) {
|
||||||
m_collapsedGroups.insert(groupName);
|
m_collapsedGroups.insert(groupName);
|
||||||
|
}
|
||||||
|
|
||||||
// Iterate through the list of instances in the group.
|
// Iterate through the list of instances in the group.
|
||||||
QJsonArray instancesArray = groupObj.value("instances").toArray();
|
QJsonArray instancesArray = groupObj.value("instances").toArray();
|
||||||
|
|
||||||
for (auto value : instancesArray) {
|
for (QJsonArray::iterator iter2 = instancesArray.begin(); iter2 != instancesArray.end(); iter2++) {
|
||||||
m_instanceGroupIndex[value.toString()] = groupName;
|
m_instanceGroupIndex[(*iter2).toString()] = groupName;
|
||||||
increaseGroupCount(groupName);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
m_groupsLoaded = true;
|
m_groupsLoaded = true;
|
||||||
|
m_groupNameCache.unite(groupSet);
|
||||||
qDebug() << "Group list loaded.";
|
qDebug() << "Group list loaded.";
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -814,7 +759,7 @@ void InstanceList::instanceDirContentsChanged(const QString& path)
|
|||||||
emit instancesChanged();
|
emit instancesChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
void InstanceList::on_InstFolderChanged([[maybe_unused]] const Setting& setting, QVariant value)
|
void InstanceList::on_InstFolderChanged(const Setting& setting, QVariant value)
|
||||||
{
|
{
|
||||||
QString newInstDir = QDir(value.toString()).canonicalPath();
|
QString newInstDir = QDir(value.toString()).canonicalPath();
|
||||||
if (newInstDir != m_instDir) {
|
if (newInstDir != m_instDir) {
|
||||||
@ -842,25 +787,20 @@ class InstanceStaging : public Task {
|
|||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
const unsigned minBackoff = 1;
|
const unsigned minBackoff = 1;
|
||||||
const unsigned maxBackoff = 16;
|
const unsigned maxBackoff = 16;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
InstanceStaging(InstanceList* parent, InstanceTask* child, QString stagingPath, InstanceName const& instanceName, QString groupName)
|
InstanceStaging(InstanceList* parent, InstanceTask* child, QString stagingPath, InstanceName const& instanceName, QString groupName)
|
||||||
: m_parent(parent)
|
: m_parent(parent), backoff(minBackoff, maxBackoff), m_stagingPath(std::move(stagingPath)), m_instance_name(std::move(instanceName)), m_groupName(std::move(groupName))
|
||||||
, backoff(minBackoff, maxBackoff)
|
|
||||||
, m_stagingPath(std::move(stagingPath))
|
|
||||||
, m_instance_name(std::move(instanceName))
|
|
||||||
, m_groupName(std::move(groupName))
|
|
||||||
{
|
{
|
||||||
m_child.reset(child);
|
m_child.reset(child);
|
||||||
connect(child, &Task::succeeded, this, &InstanceStaging::childSucceeded);
|
connect(child, &Task::succeeded, this, &InstanceStaging::childSucceded);
|
||||||
connect(child, &Task::failed, this, &InstanceStaging::childFailed);
|
connect(child, &Task::failed, this, &InstanceStaging::childFailed);
|
||||||
connect(child, &Task::aborted, this, &InstanceStaging::childAborted);
|
connect(child, &Task::aborted, this, &InstanceStaging::childAborted);
|
||||||
connect(child, &Task::abortStatusChanged, this, &InstanceStaging::setAbortable);
|
connect(child, &Task::abortStatusChanged, this, &InstanceStaging::setAbortable);
|
||||||
connect(child, &Task::status, this, &InstanceStaging::setStatus);
|
connect(child, &Task::status, this, &InstanceStaging::setStatus);
|
||||||
connect(child, &Task::details, this, &InstanceStaging::setDetails);
|
connect(child, &Task::details, this, &InstanceStaging::setDetails);
|
||||||
connect(child, &Task::progress, this, &InstanceStaging::setProgress);
|
connect(child, &Task::progress, this, &InstanceStaging::setProgress);
|
||||||
connect(child, &Task::stepProgress, this, &InstanceStaging::propagateStepProgress);
|
connect(child, &Task::stepProgress, this, &InstanceStaging::propogateStepProgress);
|
||||||
connect(&m_backoffTimer, &QTimer::timeout, this, &InstanceStaging::childSucceeded);
|
connect(&m_backoffTimer, &QTimer::timeout, this, &InstanceStaging::childSucceded);
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual ~InstanceStaging(){};
|
virtual ~InstanceStaging(){};
|
||||||
@ -875,17 +815,21 @@ class InstanceStaging : public Task {
|
|||||||
|
|
||||||
return Task::abort();
|
return Task::abort();
|
||||||
}
|
}
|
||||||
bool canAbort() const override { return (m_child && m_child->canAbort()); }
|
bool canAbort() const override
|
||||||
|
{
|
||||||
|
return (m_child && m_child->canAbort());
|
||||||
|
}
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
virtual void executeTask() override { m_child->start(); }
|
virtual void executeTask() override { m_child->start(); }
|
||||||
QStringList warnings() const override { return m_child->warnings(); }
|
QStringList warnings() const override { return m_child->warnings(); }
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void childSucceeded()
|
void childSucceded()
|
||||||
{
|
{
|
||||||
unsigned sleepTime = backoff();
|
unsigned sleepTime = backoff();
|
||||||
if (m_parent->commitStagedInstance(m_stagingPath, m_instance_name, m_groupName, *m_child.get())) {
|
if (m_parent->commitStagedInstance(m_stagingPath, m_instance_name, m_groupName, *m_child.get()))
|
||||||
|
{
|
||||||
emitSucceeded();
|
emitSucceeded();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -903,10 +847,13 @@ class InstanceStaging : public Task {
|
|||||||
emitFailed(reason);
|
emitFailed(reason);
|
||||||
}
|
}
|
||||||
|
|
||||||
void childAborted() { emitAborted(); }
|
void childAborted()
|
||||||
|
{
|
||||||
|
emitAborted();
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
InstanceList* m_parent;
|
InstanceList * m_parent;
|
||||||
/*
|
/*
|
||||||
* WHY: the whole reason why this uses an exponential backoff retry scheme is antivirus on Windows.
|
* WHY: the whole reason why this uses an exponential backoff retry scheme is antivirus on Windows.
|
||||||
* Basically, it starts messing things up while the launcher is extracting/creating instances
|
* Basically, it starts messing things up while the launcher is extracting/creating instances
|
||||||
@ -945,14 +892,8 @@ QString InstanceList::getStagedInstancePath()
|
|||||||
return path;
|
return path;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool InstanceList::commitStagedInstance(const QString& path,
|
bool InstanceList::commitStagedInstance(const QString& path, InstanceName const& instanceName, const QString& groupName, InstanceTask const& commiting)
|
||||||
InstanceName const& instanceName,
|
|
||||||
QString groupName,
|
|
||||||
InstanceTask const& commiting)
|
|
||||||
{
|
{
|
||||||
if (groupName.isEmpty() && !groupName.isNull())
|
|
||||||
groupName = QString();
|
|
||||||
|
|
||||||
QDir dir;
|
QDir dir;
|
||||||
QString instID;
|
QString instID;
|
||||||
InstancePtr inst;
|
InstancePtr inst;
|
||||||
@ -983,7 +924,7 @@ bool InstanceList::commitStagedInstance(const QString& path,
|
|||||||
}
|
}
|
||||||
|
|
||||||
m_instanceGroupIndex[instID] = groupName;
|
m_instanceGroupIndex[instID] = groupName;
|
||||||
increaseGroupCount(groupName);
|
m_groupNameCache.insert(groupName);
|
||||||
}
|
}
|
||||||
|
|
||||||
instanceSet.insert(instID);
|
instanceSet.insert(instID);
|
||||||
|
@ -1,46 +1,26 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
/* Copyright 2013-2021 MultiMC Contributors
|
||||||
/*
|
|
||||||
* Prism Launcher - Minecraft Launcher
|
|
||||||
* Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me>
|
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* it under the terms of the GNU General Public License as published by
|
* you may not use this file except in compliance with the License.
|
||||||
* the Free Software Foundation, version 3.
|
* You may obtain a copy of the License at
|
||||||
*
|
*
|
||||||
* This program is distributed in the hope that it will be useful,
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
* 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
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
*
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
* This file incorporates work covered by the following copyright and
|
* See the License for the specific language governing permissions and
|
||||||
* permission notice:
|
* limitations under the License.
|
||||||
*
|
|
||||||
* 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
|
#pragma once
|
||||||
|
|
||||||
#include <QAbstractListModel>
|
|
||||||
#include <QList>
|
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QPair>
|
#include <QAbstractListModel>
|
||||||
#include <QSet>
|
#include <QSet>
|
||||||
|
#include <QList>
|
||||||
#include <QStack>
|
#include <QStack>
|
||||||
|
#include <QPair>
|
||||||
|
|
||||||
#include "BaseInstance.h"
|
#include "BaseInstance.h"
|
||||||
|
|
||||||
@ -52,9 +32,21 @@ using InstanceId = QString;
|
|||||||
using GroupId = QString;
|
using GroupId = QString;
|
||||||
using InstanceLocator = std::pair<InstancePtr, int>;
|
using InstanceLocator = std::pair<InstancePtr, int>;
|
||||||
|
|
||||||
enum class InstCreateError { NoCreateError = 0, NoSuchVersion, UnknownCreateError, InstExists, CantCreateDir };
|
enum class InstCreateError
|
||||||
|
{
|
||||||
|
NoCreateError = 0,
|
||||||
|
NoSuchVersion,
|
||||||
|
UnknownCreateError,
|
||||||
|
InstExists,
|
||||||
|
CantCreateDir
|
||||||
|
};
|
||||||
|
|
||||||
enum class GroupsState { NotLoaded, Steady, Dirty };
|
enum class GroupsState
|
||||||
|
{
|
||||||
|
NotLoaded,
|
||||||
|
Steady,
|
||||||
|
Dirty
|
||||||
|
};
|
||||||
|
|
||||||
struct TrashHistoryItem {
|
struct TrashHistoryItem {
|
||||||
QString id;
|
QString id;
|
||||||
@ -63,36 +55,48 @@ struct TrashHistoryItem {
|
|||||||
QString groupName;
|
QString groupName;
|
||||||
};
|
};
|
||||||
|
|
||||||
class InstanceList : public QAbstractListModel {
|
class InstanceList : public QAbstractListModel
|
||||||
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit InstanceList(SettingsObjectPtr settings, const QString& instDir, QObject* parent = 0);
|
explicit InstanceList(SettingsObjectPtr settings, const QString & instDir, QObject *parent = 0);
|
||||||
virtual ~InstanceList();
|
virtual ~InstanceList();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
QModelIndex index(int row, int column = 0, const QModelIndex& parent = QModelIndex()) const override;
|
QModelIndex index(int row, int column = 0, const QModelIndex &parent = QModelIndex()) const override;
|
||||||
int rowCount(const QModelIndex& parent = QModelIndex()) const override;
|
int rowCount(const QModelIndex &parent = QModelIndex()) const override;
|
||||||
QVariant data(const QModelIndex& index, int role) const override;
|
QVariant data(const QModelIndex &index, int role) const override;
|
||||||
Qt::ItemFlags flags(const QModelIndex& index) const override;
|
Qt::ItemFlags flags(const QModelIndex &index) const override;
|
||||||
|
|
||||||
bool setData(const QModelIndex& index, const QVariant& value, int role) override;
|
bool setData(const QModelIndex & index, const QVariant & value, int role) override;
|
||||||
|
|
||||||
enum AdditionalRoles {
|
enum AdditionalRoles
|
||||||
|
{
|
||||||
GroupRole = Qt::UserRole,
|
GroupRole = Qt::UserRole,
|
||||||
InstancePointerRole = 0x34B1CB48, ///< Return pointer to real instance
|
InstancePointerRole = 0x34B1CB48, ///< Return pointer to real instance
|
||||||
InstanceIDRole = 0x34B1CB49 ///< Return id if the instance
|
InstanceIDRole = 0x34B1CB49 ///< Return id if the instance
|
||||||
};
|
};
|
||||||
/*!
|
/*!
|
||||||
* \brief Error codes returned by functions in the InstanceList class.
|
* \brief Error codes returned by functions in the InstanceList class.
|
||||||
* NoError Indicates that no error occurred.
|
* NoError Indicates that no error occurred.
|
||||||
* UnknownError indicates that an unspecified error occurred.
|
* UnknownError indicates that an unspecified error occurred.
|
||||||
*/
|
*/
|
||||||
enum InstListError { NoError = 0, UnknownError };
|
enum InstListError
|
||||||
|
{
|
||||||
|
NoError = 0,
|
||||||
|
UnknownError
|
||||||
|
};
|
||||||
|
|
||||||
InstancePtr at(int i) const { return m_instances.at(i); }
|
InstancePtr at(int i) const
|
||||||
|
{
|
||||||
|
return m_instances.at(i);
|
||||||
|
}
|
||||||
|
|
||||||
int count() const { return m_instances.count(); }
|
int count() const
|
||||||
|
{
|
||||||
|
return m_instances.count();
|
||||||
|
}
|
||||||
|
|
||||||
InstListError loadList();
|
InstListError loadList();
|
||||||
void saveNow();
|
void saveNow();
|
||||||
@ -101,22 +105,21 @@ class InstanceList : public QAbstractListModel {
|
|||||||
InstancePtr getInstanceById(QString id) const;
|
InstancePtr getInstanceById(QString id) const;
|
||||||
/* O(n) */
|
/* O(n) */
|
||||||
InstancePtr getInstanceByManagedName(const QString& managed_name) const;
|
InstancePtr getInstanceByManagedName(const QString& managed_name) const;
|
||||||
QModelIndex getInstanceIndexById(const QString& id) const;
|
QModelIndex getInstanceIndexById(const QString &id) const;
|
||||||
QStringList getGroups();
|
QStringList getGroups();
|
||||||
bool isGroupCollapsed(const QString& groupName);
|
bool isGroupCollapsed(const QString &groupName);
|
||||||
|
|
||||||
GroupId getInstanceGroup(const InstanceId& id) const;
|
GroupId getInstanceGroup(const InstanceId & id) const;
|
||||||
void setInstanceGroup(const InstanceId& id, GroupId name);
|
void setInstanceGroup(const InstanceId & id, const GroupId& name);
|
||||||
|
|
||||||
void deleteGroup(const GroupId& name);
|
void deleteGroup(const GroupId & name);
|
||||||
void renameGroup(const GroupId& src, const GroupId& dst);
|
bool trashInstance(const InstanceId &id);
|
||||||
bool trashInstance(const InstanceId& id);
|
|
||||||
bool trashedSomething();
|
bool trashedSomething();
|
||||||
void undoTrashInstance();
|
void undoTrashInstance();
|
||||||
void deleteInstance(const InstanceId& id);
|
void deleteInstance(const InstanceId & id);
|
||||||
|
|
||||||
// Wrap an instance creation task in some more task machinery and make it ready to be used
|
// Wrap an instance creation task in some more task machinery and make it ready to be used
|
||||||
Task* wrapInstanceTask(InstanceTask* task);
|
Task * wrapInstanceTask(InstanceTask * task);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new empty staging area for instance creation and @return a path/key top commit it later.
|
* Create a new empty staging area for instance creation and @return a path/key top commit it later.
|
||||||
@ -130,13 +133,13 @@ class InstanceList : public QAbstractListModel {
|
|||||||
* should_override is used when another similar instance already exists, and we want to override it
|
* should_override is used when another similar instance already exists, and we want to override it
|
||||||
* - for instance, when updating it.
|
* - for instance, when updating it.
|
||||||
*/
|
*/
|
||||||
bool commitStagedInstance(const QString& keyPath, const InstanceName& instanceName, QString groupName, const InstanceTask&);
|
bool commitStagedInstance(const QString& keyPath, const InstanceName& instanceName, const QString& groupName, const InstanceTask&);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Destroy a previously created staging area given by @keyPath - used when creation fails.
|
* Destroy a previously created staging area given by @keyPath - used when creation fails.
|
||||||
* Used by instance manipulation tasks.
|
* Used by instance manipulation tasks.
|
||||||
*/
|
*/
|
||||||
bool destroyStagingPath(const QString& keyPath);
|
bool destroyStagingPath(const QString & keyPath);
|
||||||
|
|
||||||
int getTotalPlayTime();
|
int getTotalPlayTime();
|
||||||
|
|
||||||
@ -144,55 +147,51 @@ class InstanceList : public QAbstractListModel {
|
|||||||
|
|
||||||
Qt::DropActions supportedDropActions() const override;
|
Qt::DropActions supportedDropActions() const override;
|
||||||
|
|
||||||
bool canDropMimeData(const QMimeData* data, Qt::DropAction action, int row, int column, const QModelIndex& parent) const override;
|
bool canDropMimeData(const QMimeData * data, Qt::DropAction action, int row, int column, const QModelIndex & parent) const override;
|
||||||
|
|
||||||
bool dropMimeData(const QMimeData* data, Qt::DropAction action, int row, int column, const QModelIndex& parent) override;
|
bool dropMimeData(const QMimeData * data, Qt::DropAction action, int row, int column, const QModelIndex & parent) override;
|
||||||
|
|
||||||
QStringList mimeTypes() const override;
|
QStringList mimeTypes() const override;
|
||||||
QMimeData* mimeData(const QModelIndexList& indexes) const override;
|
QMimeData *mimeData(const QModelIndexList &indexes) const override;
|
||||||
|
|
||||||
QStringList getLinkedInstancesById(const QString& id) const;
|
QStringList getLinkedInstancesById(const QString &id) const;
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void dataIsInvalid();
|
void dataIsInvalid();
|
||||||
void instancesChanged();
|
void instancesChanged();
|
||||||
void instanceSelectRequest(QString instanceId);
|
void instanceSelectRequest(QString instanceId);
|
||||||
void groupsChanged(QSet<QString> groups);
|
void groupsChanged(QSet<QString> groups);
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void on_InstFolderChanged(const Setting& setting, QVariant value);
|
void on_InstFolderChanged(const Setting &setting, QVariant value);
|
||||||
void on_GroupStateChanged(const QString& group, bool collapsed);
|
void on_GroupStateChanged(const QString &group, bool collapsed);
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void propertiesChanged(BaseInstance* inst);
|
void propertiesChanged(BaseInstance *inst);
|
||||||
void providerUpdated();
|
void providerUpdated();
|
||||||
void instanceDirContentsChanged(const QString& path);
|
void instanceDirContentsChanged(const QString &path);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
int getInstIndex(BaseInstance* inst) const;
|
int getInstIndex(BaseInstance *inst) const;
|
||||||
void updateTotalPlayTime();
|
void updateTotalPlayTime();
|
||||||
void suspendWatch();
|
void suspendWatch();
|
||||||
void resumeWatch();
|
void resumeWatch();
|
||||||
void add(const QList<InstancePtr>& list);
|
void add(const QList<InstancePtr> &list);
|
||||||
void loadGroupList();
|
void loadGroupList();
|
||||||
void saveGroupList();
|
void saveGroupList();
|
||||||
QList<InstanceId> discoverInstances();
|
QList<InstanceId> discoverInstances();
|
||||||
InstancePtr loadInstance(const InstanceId& id);
|
InstancePtr loadInstance(const InstanceId& id);
|
||||||
|
|
||||||
void increaseGroupCount(const QString& group);
|
private:
|
||||||
void decreaseGroupCount(const QString& group);
|
|
||||||
|
|
||||||
private:
|
|
||||||
int m_watchLevel = 0;
|
int m_watchLevel = 0;
|
||||||
int totalPlayTime = 0;
|
int totalPlayTime = 0;
|
||||||
bool m_dirty = false;
|
bool m_dirty = false;
|
||||||
QList<InstancePtr> m_instances;
|
QList<InstancePtr> m_instances;
|
||||||
// id -> refs
|
QSet<QString> m_groupNameCache;
|
||||||
QMap<QString, int> m_groupNameCache;
|
|
||||||
|
|
||||||
SettingsObjectPtr m_globalSettings;
|
SettingsObjectPtr m_globalSettings;
|
||||||
QString m_instDir;
|
QString m_instDir;
|
||||||
QFileSystemWatcher* m_watcher;
|
QFileSystemWatcher * m_watcher;
|
||||||
// FIXME: this is so inefficient that looking at it is almost painful.
|
// FIXME: this is so inefficient that looking at it is almost painful.
|
||||||
QSet<QString> m_collapsedGroups;
|
QSet<QString> m_collapsedGroups;
|
||||||
QMap<InstanceId, GroupId> m_instanceGroupIndex;
|
QMap<InstanceId, GroupId> m_instanceGroupIndex;
|
||||||
|
@ -1,31 +1,36 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <FileSystem.h>
|
|
||||||
#include "minecraft/MinecraftInstance.h"
|
#include "minecraft/MinecraftInstance.h"
|
||||||
|
#include <FileSystem.h>
|
||||||
#include "ui/pages/BasePage.h"
|
#include "ui/pages/BasePage.h"
|
||||||
#include "ui/pages/BasePageProvider.h"
|
#include "ui/pages/BasePageProvider.h"
|
||||||
#include "ui/pages/instance/InstanceSettingsPage.h"
|
|
||||||
#include "ui/pages/instance/LogPage.h"
|
#include "ui/pages/instance/LogPage.h"
|
||||||
|
#include "ui/pages/instance/VersionPage.h"
|
||||||
#include "ui/pages/instance/ManagedPackPage.h"
|
#include "ui/pages/instance/ManagedPackPage.h"
|
||||||
#include "ui/pages/instance/ModFolderPage.h"
|
#include "ui/pages/instance/ModFolderPage.h"
|
||||||
#include "ui/pages/instance/NotesPage.h"
|
|
||||||
#include "ui/pages/instance/OtherLogsPage.h"
|
|
||||||
#include "ui/pages/instance/ResourcePackPage.h"
|
#include "ui/pages/instance/ResourcePackPage.h"
|
||||||
#include "ui/pages/instance/ScreenshotsPage.h"
|
|
||||||
#include "ui/pages/instance/ServersPage.h"
|
|
||||||
#include "ui/pages/instance/ShaderPackPage.h"
|
|
||||||
#include "ui/pages/instance/TexturePackPage.h"
|
#include "ui/pages/instance/TexturePackPage.h"
|
||||||
#include "ui/pages/instance/VersionPage.h"
|
#include "ui/pages/instance/ShaderPackPage.h"
|
||||||
|
#include "ui/pages/instance/NotesPage.h"
|
||||||
|
#include "ui/pages/instance/ScreenshotsPage.h"
|
||||||
|
#include "ui/pages/instance/InstanceSettingsPage.h"
|
||||||
|
#include "ui/pages/instance/OtherLogsPage.h"
|
||||||
#include "ui/pages/instance/WorldListPage.h"
|
#include "ui/pages/instance/WorldListPage.h"
|
||||||
|
#include "ui/pages/instance/ServersPage.h"
|
||||||
|
#include "ui/pages/instance/GameOptionsPage.h"
|
||||||
|
|
||||||
class InstancePageProvider : protected QObject, public BasePageProvider {
|
class InstancePageProvider : public QObject, public BasePageProvider
|
||||||
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
explicit InstancePageProvider(InstancePtr parent) { inst = parent; }
|
explicit InstancePageProvider(InstancePtr parent)
|
||||||
|
|
||||||
virtual ~InstancePageProvider(){};
|
|
||||||
virtual QList<BasePage*> getPages() override
|
|
||||||
{
|
{
|
||||||
QList<BasePage*> values;
|
inst = parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual ~InstancePageProvider() {};
|
||||||
|
virtual QList<BasePage *> getPages() override
|
||||||
|
{
|
||||||
|
QList<BasePage *> values;
|
||||||
values.append(new LogPage(inst));
|
values.append(new LogPage(inst));
|
||||||
std::shared_ptr<MinecraftInstance> onesix = std::dynamic_pointer_cast<MinecraftInstance>(inst);
|
std::shared_ptr<MinecraftInstance> onesix = std::dynamic_pointer_cast<MinecraftInstance>(inst);
|
||||||
values.append(new VersionPage(onesix.get()));
|
values.append(new VersionPage(onesix.get()));
|
||||||
@ -45,14 +50,18 @@ class InstancePageProvider : protected QObject, public BasePageProvider {
|
|||||||
values.append(new ScreenshotsPage(FS::PathCombine(onesix->gameRoot(), "screenshots")));
|
values.append(new ScreenshotsPage(FS::PathCombine(onesix->gameRoot(), "screenshots")));
|
||||||
values.append(new InstanceSettingsPage(onesix.get()));
|
values.append(new InstanceSettingsPage(onesix.get()));
|
||||||
auto logMatcher = inst->getLogFileMatcher();
|
auto logMatcher = inst->getLogFileMatcher();
|
||||||
if (logMatcher) {
|
if(logMatcher)
|
||||||
|
{
|
||||||
values.append(new OtherLogsPage(inst->getLogFileRoot(), logMatcher));
|
values.append(new OtherLogsPage(inst->getLogFileRoot(), logMatcher));
|
||||||
}
|
}
|
||||||
return values;
|
return values;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual QString dialogTitle() override { return tr("Edit Instance (%1)").arg(inst->name()); }
|
virtual QString dialogTitle() override
|
||||||
|
{
|
||||||
protected:
|
return tr("Edit Instance (%1)").arg(inst->name());
|
||||||
|
}
|
||||||
|
protected:
|
||||||
InstancePtr inst;
|
InstancePtr inst;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -18,14 +18,13 @@ InstanceNameChange askForChangingInstanceName(QWidget* parent, const QString& ol
|
|||||||
return InstanceNameChange::ShouldKeep;
|
return InstanceNameChange::ShouldKeep;
|
||||||
}
|
}
|
||||||
|
|
||||||
ShouldUpdate askIfShouldUpdate(QWidget* parent, QString original_version_name)
|
ShouldUpdate askIfShouldUpdate(QWidget *parent, QString original_version_name)
|
||||||
{
|
{
|
||||||
auto info = CustomMessageBox::selectable(
|
auto info = CustomMessageBox::selectable(
|
||||||
parent, QObject::tr("Similar modpack was found!"),
|
parent, QObject::tr("Similar modpack was found!"),
|
||||||
QObject::tr(
|
QObject::tr("One or more of your instances are from this same modpack%1. Do you want to create a "
|
||||||
"One or more of your instances are from this same modpack%1. Do you want to create a "
|
"separate instance, or update the existing one?\n\nNOTE: Make sure you made a backup of your important instance data before "
|
||||||
"separate instance, or update the existing one?\n\nNOTE: Make sure you made a backup of your important instance data before "
|
"updating, as worlds can be corrupted and some configuration may be lost (due to pack overrides).")
|
||||||
"updating, as worlds can be corrupted and some configuration may be lost (due to pack overrides).")
|
|
||||||
.arg(original_version_name),
|
.arg(original_version_name),
|
||||||
QMessageBox::Information, QMessageBox::Ok | QMessageBox::Reset | QMessageBox::Abort);
|
QMessageBox::Information, QMessageBox::Ok | QMessageBox::Reset | QMessageBox::Abort);
|
||||||
info->setButtonText(QMessageBox::Ok, QObject::tr("Update existing instance"));
|
info->setButtonText(QMessageBox::Ok, QObject::tr("Update existing instance"));
|
||||||
@ -39,6 +38,7 @@ ShouldUpdate askIfShouldUpdate(QWidget* parent, QString original_version_name)
|
|||||||
if (info->clickedButton() == info->button(QMessageBox::Abort))
|
if (info->clickedButton() == info->button(QMessageBox::Abort))
|
||||||
return ShouldUpdate::SkipUpdating;
|
return ShouldUpdate::SkipUpdating;
|
||||||
return ShouldUpdate::Cancel;
|
return ShouldUpdate::Cancel;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
QString InstanceName::name() const
|
QString InstanceName::name() const
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
@ -39,39 +39,43 @@
|
|||||||
|
|
||||||
#include <QRegularExpression>
|
#include <QRegularExpression>
|
||||||
|
|
||||||
bool JavaCommon::checkJVMArgs(QString jvmargs, QWidget* parent)
|
bool JavaCommon::checkJVMArgs(QString jvmargs, QWidget *parent)
|
||||||
{
|
{
|
||||||
if (jvmargs.contains("-XX:PermSize=") || jvmargs.contains(QRegularExpression("-Xm[sx]")) || jvmargs.contains("-XX-MaxHeapSize") ||
|
if (jvmargs.contains("-XX:PermSize=") || jvmargs.contains(QRegularExpression("-Xm[sx]"))
|
||||||
jvmargs.contains("-XX:InitialHeapSize")) {
|
|| jvmargs.contains("-XX-MaxHeapSize") || jvmargs.contains("-XX:InitialHeapSize"))
|
||||||
|
{
|
||||||
auto warnStr = QObject::tr(
|
auto warnStr = QObject::tr(
|
||||||
"You tried to manually set a JVM memory option (using \"-XX:PermSize\", \"-XX-MaxHeapSize\", \"-XX:InitialHeapSize\", \"-Xmx\" "
|
"You tried to manually set a JVM memory option (using \"-XX:PermSize\", \"-XX-MaxHeapSize\", \"-XX:InitialHeapSize\", \"-Xmx\" or \"-Xms\").\n"
|
||||||
"or \"-Xms\").\n"
|
|
||||||
"There are dedicated boxes for these in the settings (Java tab, in the Memory group at the top).\n"
|
"There are dedicated boxes for these in the settings (Java tab, in the Memory group at the top).\n"
|
||||||
"This message will be displayed until you remove them from the JVM arguments.");
|
"This message will be displayed until you remove them from the JVM arguments.");
|
||||||
CustomMessageBox::selectable(parent, QObject::tr("JVM arguments warning"), warnStr, QMessageBox::Warning)->exec();
|
CustomMessageBox::selectable(
|
||||||
|
parent, QObject::tr("JVM arguments warning"),
|
||||||
|
warnStr,
|
||||||
|
QMessageBox::Warning)->exec();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
// block lunacy with passing required version to the JVM
|
// block lunacy with passing required version to the JVM
|
||||||
if (jvmargs.contains(QRegularExpression("-version:.*"))) {
|
if (jvmargs.contains(QRegularExpression("-version:.*"))) {
|
||||||
auto warnStr = QObject::tr(
|
auto warnStr = QObject::tr(
|
||||||
"You tried to pass required Java version argument to the JVM (using \"-version:xxx\"). This is not safe and will not be "
|
"You tried to pass required Java version argument to the JVM (using \"-version:xxx\"). This is not safe and will not be allowed.\n"
|
||||||
"allowed.\n"
|
|
||||||
"This message will be displayed until you remove this from the JVM arguments.");
|
"This message will be displayed until you remove this from the JVM arguments.");
|
||||||
CustomMessageBox::selectable(parent, QObject::tr("JVM arguments warning"), warnStr, QMessageBox::Warning)->exec();
|
CustomMessageBox::selectable(
|
||||||
|
parent, QObject::tr("JVM arguments warning"),
|
||||||
|
warnStr,
|
||||||
|
QMessageBox::Warning)->exec();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void JavaCommon::javaWasOk(QWidget* parent, const JavaCheckResult& result)
|
void JavaCommon::javaWasOk(QWidget *parent, JavaCheckResult result)
|
||||||
{
|
{
|
||||||
QString text;
|
QString text;
|
||||||
text += QObject::tr(
|
text += QObject::tr("Java test succeeded!<br />Platform reported: %1<br />Java version "
|
||||||
"Java test succeeded!<br />Platform reported: %1<br />Java version "
|
"reported: %2<br />Java vendor "
|
||||||
"reported: %2<br />Java vendor "
|
"reported: %3<br />").arg(result.realPlatform, result.javaVersion.toString(), result.javaVendor);
|
||||||
"reported: %3<br />")
|
if (result.errorLog.size())
|
||||||
.arg(result.realPlatform, result.javaVersion.toString(), result.javaVendor);
|
{
|
||||||
if (result.errorLog.size()) {
|
|
||||||
auto htmlError = result.errorLog;
|
auto htmlError = result.errorLog;
|
||||||
htmlError.replace('\n', "<br />");
|
htmlError.replace('\n', "<br />");
|
||||||
text += QObject::tr("<br />Warnings:<br /><font color=\"orange\">%1</font>").arg(htmlError);
|
text += QObject::tr("<br />Warnings:<br /><font color=\"orange\">%1</font>").arg(htmlError);
|
||||||
@ -79,7 +83,7 @@ void JavaCommon::javaWasOk(QWidget* parent, const JavaCheckResult& result)
|
|||||||
CustomMessageBox::selectable(parent, QObject::tr("Java test success"), text, QMessageBox::Information)->show();
|
CustomMessageBox::selectable(parent, QObject::tr("Java test success"), text, QMessageBox::Information)->show();
|
||||||
}
|
}
|
||||||
|
|
||||||
void JavaCommon::javaArgsWereBad(QWidget* parent, const JavaCheckResult& result)
|
void JavaCommon::javaArgsWereBad(QWidget *parent, JavaCheckResult result)
|
||||||
{
|
{
|
||||||
auto htmlError = result.errorLog;
|
auto htmlError = result.errorLog;
|
||||||
QString text;
|
QString text;
|
||||||
@ -89,7 +93,7 @@ void JavaCommon::javaArgsWereBad(QWidget* parent, const JavaCheckResult& result)
|
|||||||
CustomMessageBox::selectable(parent, QObject::tr("Java test failure"), text, QMessageBox::Warning)->show();
|
CustomMessageBox::selectable(parent, QObject::tr("Java test failure"), text, QMessageBox::Warning)->show();
|
||||||
}
|
}
|
||||||
|
|
||||||
void JavaCommon::javaBinaryWasBad(QWidget* parent, const JavaCheckResult& result)
|
void JavaCommon::javaBinaryWasBad(QWidget *parent, JavaCheckResult result)
|
||||||
{
|
{
|
||||||
QString text;
|
QString text;
|
||||||
text += QObject::tr(
|
text += QObject::tr(
|
||||||
@ -98,7 +102,7 @@ void JavaCommon::javaBinaryWasBad(QWidget* parent, const JavaCheckResult& result
|
|||||||
CustomMessageBox::selectable(parent, QObject::tr("Java test failure"), text, QMessageBox::Warning)->show();
|
CustomMessageBox::selectable(parent, QObject::tr("Java test failure"), text, QMessageBox::Warning)->show();
|
||||||
}
|
}
|
||||||
|
|
||||||
void JavaCommon::javaCheckNotFound(QWidget* parent)
|
void JavaCommon::javaCheckNotFound(QWidget *parent)
|
||||||
{
|
{
|
||||||
QString text;
|
QString text;
|
||||||
text += QObject::tr("Java checker library could not be found. Please check your installation.");
|
text += QObject::tr("Java checker library could not be found. Please check your installation.");
|
||||||
@ -107,7 +111,8 @@ void JavaCommon::javaCheckNotFound(QWidget* parent)
|
|||||||
|
|
||||||
void JavaCommon::TestCheck::run()
|
void JavaCommon::TestCheck::run()
|
||||||
{
|
{
|
||||||
if (!JavaCommon::checkJVMArgs(m_args, m_parent)) {
|
if (!JavaCommon::checkJVMArgs(m_args, m_parent))
|
||||||
|
{
|
||||||
emit finished();
|
emit finished();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -124,7 +129,8 @@ void JavaCommon::TestCheck::run()
|
|||||||
|
|
||||||
void JavaCommon::TestCheck::checkFinished(JavaCheckResult result)
|
void JavaCommon::TestCheck::checkFinished(JavaCheckResult result)
|
||||||
{
|
{
|
||||||
if (result.validity != JavaCheckResult::Validity::Valid) {
|
if (result.validity != JavaCheckResult::Validity::Valid)
|
||||||
|
{
|
||||||
javaBinaryWasBad(m_parent, result);
|
javaBinaryWasBad(m_parent, result);
|
||||||
emit finished();
|
emit finished();
|
||||||
return;
|
return;
|
||||||
@ -135,7 +141,8 @@ void JavaCommon::TestCheck::checkFinished(JavaCheckResult result)
|
|||||||
checker->m_args = m_args;
|
checker->m_args = m_args;
|
||||||
checker->m_minMem = m_minMem;
|
checker->m_minMem = m_minMem;
|
||||||
checker->m_maxMem = m_maxMem;
|
checker->m_maxMem = m_maxMem;
|
||||||
if (result.javaVersion.requiresPermGen()) {
|
if (result.javaVersion.requiresPermGen())
|
||||||
|
{
|
||||||
checker->m_permGen = m_permGen;
|
checker->m_permGen = m_permGen;
|
||||||
}
|
}
|
||||||
checker->performCheck();
|
checker->performCheck();
|
||||||
@ -143,7 +150,8 @@ void JavaCommon::TestCheck::checkFinished(JavaCheckResult result)
|
|||||||
|
|
||||||
void JavaCommon::TestCheck::checkFinishedWithArgs(JavaCheckResult result)
|
void JavaCommon::TestCheck::checkFinishedWithArgs(JavaCheckResult result)
|
||||||
{
|
{
|
||||||
if (result.validity == JavaCheckResult::Validity::Valid) {
|
if (result.validity == JavaCheckResult::Validity::Valid)
|
||||||
|
{
|
||||||
javaWasOk(m_parent, result);
|
javaWasOk(m_parent, result);
|
||||||
emit finished();
|
emit finished();
|
||||||
return;
|
return;
|
||||||
@ -151,3 +159,4 @@ void JavaCommon::TestCheck::checkFinishedWithArgs(JavaCheckResult result)
|
|||||||
javaArgsWereBad(m_parent, result);
|
javaArgsWereBad(m_parent, result);
|
||||||
emit finished();
|
emit finished();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -6,42 +6,45 @@ class QWidget;
|
|||||||
/**
|
/**
|
||||||
* Common UI bits for the java pages to use.
|
* Common UI bits for the java pages to use.
|
||||||
*/
|
*/
|
||||||
namespace JavaCommon {
|
namespace JavaCommon
|
||||||
bool checkJVMArgs(QString args, QWidget* parent);
|
{
|
||||||
|
bool checkJVMArgs(QString args, QWidget *parent);
|
||||||
|
|
||||||
// Show a dialog saying that the Java binary was usable
|
// Show a dialog saying that the Java binary was usable
|
||||||
void javaWasOk(QWidget* parent, const JavaCheckResult& result);
|
void javaWasOk(QWidget *parent, JavaCheckResult result);
|
||||||
// Show a dialog saying that the Java binary was not usable because of bad options
|
// Show a dialog saying that the Java binary was not usable because of bad options
|
||||||
void javaArgsWereBad(QWidget* parent, const JavaCheckResult& result);
|
void javaArgsWereBad(QWidget *parent, JavaCheckResult result);
|
||||||
// Show a dialog saying that the Java binary was not usable
|
// Show a dialog saying that the Java binary was not usable
|
||||||
void javaBinaryWasBad(QWidget* parent, const JavaCheckResult& result);
|
void javaBinaryWasBad(QWidget *parent, JavaCheckResult result);
|
||||||
// Show a dialog if we couldn't find Java Checker
|
// Show a dialog if we couldn't find Java Checker
|
||||||
void javaCheckNotFound(QWidget* parent);
|
void javaCheckNotFound(QWidget *parent);
|
||||||
|
|
||||||
class TestCheck : public QObject {
|
class TestCheck : public QObject
|
||||||
Q_OBJECT
|
{
|
||||||
public:
|
Q_OBJECT
|
||||||
TestCheck(QWidget* parent, QString path, QString args, int minMem, int maxMem, int permGen)
|
public:
|
||||||
: m_parent(parent), m_path(path), m_args(args), m_minMem(minMem), m_maxMem(maxMem), m_permGen(permGen)
|
TestCheck(QWidget *parent, QString path, QString args, int minMem, int maxMem, int permGen)
|
||||||
{}
|
:m_parent(parent), m_path(path), m_args(args), m_minMem(minMem), m_maxMem(maxMem), m_permGen(permGen)
|
||||||
virtual ~TestCheck(){};
|
{
|
||||||
|
}
|
||||||
|
virtual ~TestCheck() {};
|
||||||
|
|
||||||
void run();
|
void run();
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void finished();
|
void finished();
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void checkFinished(JavaCheckResult result);
|
void checkFinished(JavaCheckResult result);
|
||||||
void checkFinishedWithArgs(JavaCheckResult result);
|
void checkFinishedWithArgs(JavaCheckResult result);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::shared_ptr<JavaChecker> checker;
|
std::shared_ptr<JavaChecker> checker;
|
||||||
QWidget* m_parent = nullptr;
|
QWidget *m_parent = nullptr;
|
||||||
QString m_path;
|
QString m_path;
|
||||||
QString m_args;
|
QString m_args;
|
||||||
int m_minMem = 0;
|
int m_minMem = 0;
|
||||||
int m_maxMem = 0;
|
int m_maxMem = 0;
|
||||||
int m_permGen = 64;
|
int m_permGen = 64;
|
||||||
};
|
};
|
||||||
} // namespace JavaCommon
|
}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
@ -37,246 +37,257 @@
|
|||||||
|
|
||||||
#include <QFile>
|
#include <QFile>
|
||||||
|
|
||||||
#include <math.h>
|
|
||||||
#include "FileSystem.h"
|
#include "FileSystem.h"
|
||||||
|
#include <math.h>
|
||||||
|
|
||||||
namespace Json {
|
namespace Json
|
||||||
void write(const QJsonDocument& doc, const QString& filename)
|
{
|
||||||
|
void write(const QJsonDocument &doc, const QString &filename)
|
||||||
{
|
{
|
||||||
FS::write(filename, doc.toJson());
|
FS::write(filename, doc.toJson());
|
||||||
}
|
}
|
||||||
void write(const QJsonObject& object, const QString& filename)
|
void write(const QJsonObject &object, const QString &filename)
|
||||||
{
|
{
|
||||||
write(QJsonDocument(object), filename);
|
write(QJsonDocument(object), filename);
|
||||||
}
|
}
|
||||||
void write(const QJsonArray& array, const QString& filename)
|
void write(const QJsonArray &array, const QString &filename)
|
||||||
{
|
{
|
||||||
write(QJsonDocument(array), filename);
|
write(QJsonDocument(array), filename);
|
||||||
}
|
}
|
||||||
|
|
||||||
QByteArray toText(const QJsonObject& obj)
|
QByteArray toText(const QJsonObject &obj)
|
||||||
{
|
{
|
||||||
return QJsonDocument(obj).toJson(QJsonDocument::Compact);
|
return QJsonDocument(obj).toJson(QJsonDocument::Compact);
|
||||||
}
|
}
|
||||||
QByteArray toText(const QJsonArray& array)
|
QByteArray toText(const QJsonArray &array)
|
||||||
{
|
{
|
||||||
return QJsonDocument(array).toJson(QJsonDocument::Compact);
|
return QJsonDocument(array).toJson(QJsonDocument::Compact);
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool isBinaryJson(const QByteArray& data)
|
static bool isBinaryJson(const QByteArray &data)
|
||||||
{
|
{
|
||||||
decltype(QJsonDocument::BinaryFormatTag) tag = QJsonDocument::BinaryFormatTag;
|
decltype(QJsonDocument::BinaryFormatTag) tag = QJsonDocument::BinaryFormatTag;
|
||||||
return memcmp(data.constData(), &tag, sizeof(QJsonDocument::BinaryFormatTag)) == 0;
|
return memcmp(data.constData(), &tag, sizeof(QJsonDocument::BinaryFormatTag)) == 0;
|
||||||
}
|
}
|
||||||
QJsonDocument requireDocument(const QByteArray& data, const QString& what)
|
QJsonDocument requireDocument(const QByteArray &data, const QString &what)
|
||||||
{
|
{
|
||||||
if (isBinaryJson(data)) {
|
if (isBinaryJson(data))
|
||||||
|
{
|
||||||
// FIXME: Is this needed?
|
// FIXME: Is this needed?
|
||||||
throw JsonException(what + ": Invalid JSON. Binary JSON unsupported");
|
throw JsonException(what + ": Invalid JSON. Binary JSON unsupported");
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
QJsonParseError error;
|
QJsonParseError error;
|
||||||
QJsonDocument doc = QJsonDocument::fromJson(data, &error);
|
QJsonDocument doc = QJsonDocument::fromJson(data, &error);
|
||||||
if (error.error != QJsonParseError::NoError) {
|
if (error.error != QJsonParseError::NoError)
|
||||||
|
{
|
||||||
throw JsonException(what + ": Error parsing JSON: " + error.errorString());
|
throw JsonException(what + ": Error parsing JSON: " + error.errorString());
|
||||||
}
|
}
|
||||||
return doc;
|
return doc;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
QJsonDocument requireDocument(const QString& filename, const QString& what)
|
QJsonDocument requireDocument(const QString &filename, const QString &what)
|
||||||
{
|
{
|
||||||
return requireDocument(FS::read(filename), what);
|
return requireDocument(FS::read(filename), what);
|
||||||
}
|
}
|
||||||
QJsonObject requireObject(const QJsonDocument& doc, const QString& what)
|
QJsonObject requireObject(const QJsonDocument &doc, const QString &what)
|
||||||
{
|
{
|
||||||
if (!doc.isObject()) {
|
if (!doc.isObject())
|
||||||
|
{
|
||||||
throw JsonException(what + " is not an object");
|
throw JsonException(what + " is not an object");
|
||||||
}
|
}
|
||||||
return doc.object();
|
return doc.object();
|
||||||
}
|
}
|
||||||
QJsonArray requireArray(const QJsonDocument& doc, const QString& what)
|
QJsonArray requireArray(const QJsonDocument &doc, const QString &what)
|
||||||
{
|
{
|
||||||
if (!doc.isArray()) {
|
if (!doc.isArray())
|
||||||
|
{
|
||||||
throw JsonException(what + " is not an array");
|
throw JsonException(what + " is not an array");
|
||||||
}
|
}
|
||||||
return doc.array();
|
return doc.array();
|
||||||
}
|
}
|
||||||
|
|
||||||
void writeString(QJsonObject& to, const QString& key, const QString& value)
|
void writeString(QJsonObject &to, const QString &key, const QString &value)
|
||||||
{
|
{
|
||||||
if (!value.isEmpty()) {
|
if (!value.isEmpty())
|
||||||
|
{
|
||||||
to.insert(key, value);
|
to.insert(key, value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void writeStringList(QJsonObject& to, const QString& key, const QStringList& values)
|
void writeStringList(QJsonObject &to, const QString &key, const QStringList &values)
|
||||||
{
|
{
|
||||||
if (!values.isEmpty()) {
|
if (!values.isEmpty())
|
||||||
|
{
|
||||||
QJsonArray array;
|
QJsonArray array;
|
||||||
for (auto value : values) {
|
for(auto value: values)
|
||||||
|
{
|
||||||
array.append(value);
|
array.append(value);
|
||||||
}
|
}
|
||||||
to.insert(key, array);
|
to.insert(key, array);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template<>
|
||||||
QJsonValue toJson<QUrl>(const QUrl& url)
|
QJsonValue toJson<QUrl>(const QUrl &url)
|
||||||
{
|
{
|
||||||
return QJsonValue(url.toString(QUrl::FullyEncoded));
|
return QJsonValue(url.toString(QUrl::FullyEncoded));
|
||||||
}
|
}
|
||||||
template <>
|
template<>
|
||||||
QJsonValue toJson<QByteArray>(const QByteArray& data)
|
QJsonValue toJson<QByteArray>(const QByteArray &data)
|
||||||
{
|
{
|
||||||
return QJsonValue(QString::fromLatin1(data.toHex()));
|
return QJsonValue(QString::fromLatin1(data.toHex()));
|
||||||
}
|
}
|
||||||
template <>
|
template<>
|
||||||
QJsonValue toJson<QDateTime>(const QDateTime& datetime)
|
QJsonValue toJson<QDateTime>(const QDateTime &datetime)
|
||||||
{
|
{
|
||||||
return QJsonValue(datetime.toString(Qt::ISODate));
|
return QJsonValue(datetime.toString(Qt::ISODate));
|
||||||
}
|
}
|
||||||
template <>
|
template<>
|
||||||
QJsonValue toJson<QDir>(const QDir& dir)
|
QJsonValue toJson<QDir>(const QDir &dir)
|
||||||
{
|
{
|
||||||
return QDir::current().relativeFilePath(dir.absolutePath());
|
return QDir::current().relativeFilePath(dir.absolutePath());
|
||||||
}
|
}
|
||||||
template <>
|
template<>
|
||||||
QJsonValue toJson<QUuid>(const QUuid& uuid)
|
QJsonValue toJson<QUuid>(const QUuid &uuid)
|
||||||
{
|
{
|
||||||
return uuid.toString();
|
return uuid.toString();
|
||||||
}
|
}
|
||||||
template <>
|
template<>
|
||||||
QJsonValue toJson<QVariant>(const QVariant& variant)
|
QJsonValue toJson<QVariant>(const QVariant &variant)
|
||||||
{
|
{
|
||||||
return QJsonValue::fromVariant(variant);
|
return QJsonValue::fromVariant(variant);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
|
||||||
QByteArray requireIsType<QByteArray>(const QJsonValue& value, const QString& what)
|
template<> QByteArray requireIsType<QByteArray>(const QJsonValue &value, const QString &what)
|
||||||
{
|
{
|
||||||
const QString string = ensureIsType<QString>(value, what);
|
const QString string = ensureIsType<QString>(value, what);
|
||||||
// ensure that the string can be safely cast to Latin1
|
// ensure that the string can be safely cast to Latin1
|
||||||
if (string != QString::fromLatin1(string.toLatin1())) {
|
if (string != QString::fromLatin1(string.toLatin1()))
|
||||||
|
{
|
||||||
throw JsonException(what + " is not encodable as Latin1");
|
throw JsonException(what + " is not encodable as Latin1");
|
||||||
}
|
}
|
||||||
return QByteArray::fromHex(string.toLatin1());
|
return QByteArray::fromHex(string.toLatin1());
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template<> QJsonArray requireIsType<QJsonArray>(const QJsonValue &value, const QString &what)
|
||||||
QJsonArray requireIsType<QJsonArray>(const QJsonValue& value, const QString& what)
|
|
||||||
{
|
{
|
||||||
if (!value.isArray()) {
|
if (!value.isArray())
|
||||||
|
{
|
||||||
throw JsonException(what + " is not an array");
|
throw JsonException(what + " is not an array");
|
||||||
}
|
}
|
||||||
return value.toArray();
|
return value.toArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
|
||||||
QString requireIsType<QString>(const QJsonValue& value, const QString& what)
|
template<> QString requireIsType<QString>(const QJsonValue &value, const QString &what)
|
||||||
{
|
{
|
||||||
if (!value.isString()) {
|
if (!value.isString())
|
||||||
|
{
|
||||||
throw JsonException(what + " is not a string");
|
throw JsonException(what + " is not a string");
|
||||||
}
|
}
|
||||||
return value.toString();
|
return value.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template<> bool requireIsType<bool>(const QJsonValue &value, const QString &what)
|
||||||
bool requireIsType<bool>(const QJsonValue& value, const QString& what)
|
|
||||||
{
|
{
|
||||||
if (!value.isBool()) {
|
if (!value.isBool())
|
||||||
|
{
|
||||||
throw JsonException(what + " is not a bool");
|
throw JsonException(what + " is not a bool");
|
||||||
}
|
}
|
||||||
return value.toBool();
|
return value.toBool();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template<> double requireIsType<double>(const QJsonValue &value, const QString &what)
|
||||||
double requireIsType<double>(const QJsonValue& value, const QString& what)
|
|
||||||
{
|
{
|
||||||
if (!value.isDouble()) {
|
if (!value.isDouble())
|
||||||
|
{
|
||||||
throw JsonException(what + " is not a double");
|
throw JsonException(what + " is not a double");
|
||||||
}
|
}
|
||||||
return value.toDouble();
|
return value.toDouble();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template<> int requireIsType<int>(const QJsonValue &value, const QString &what)
|
||||||
int requireIsType<int>(const QJsonValue& value, const QString& what)
|
|
||||||
{
|
{
|
||||||
const double doubl = requireIsType<double>(value, what);
|
const double doubl = requireIsType<double>(value, what);
|
||||||
if (fmod(doubl, 1) != 0) {
|
if (fmod(doubl, 1) != 0)
|
||||||
|
{
|
||||||
throw JsonException(what + " is not an integer");
|
throw JsonException(what + " is not an integer");
|
||||||
}
|
}
|
||||||
return int(doubl);
|
return int(doubl);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template<> QDateTime requireIsType<QDateTime>(const QJsonValue &value, const QString &what)
|
||||||
QDateTime requireIsType<QDateTime>(const QJsonValue& value, const QString& what)
|
|
||||||
{
|
{
|
||||||
const QString string = requireIsType<QString>(value, what);
|
const QString string = requireIsType<QString>(value, what);
|
||||||
const QDateTime datetime = QDateTime::fromString(string, Qt::ISODate);
|
const QDateTime datetime = QDateTime::fromString(string, Qt::ISODate);
|
||||||
if (!datetime.isValid()) {
|
if (!datetime.isValid())
|
||||||
|
{
|
||||||
throw JsonException(what + " is not a ISO formatted date/time value");
|
throw JsonException(what + " is not a ISO formatted date/time value");
|
||||||
}
|
}
|
||||||
return datetime;
|
return datetime;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template<> QUrl requireIsType<QUrl>(const QJsonValue &value, const QString &what)
|
||||||
QUrl requireIsType<QUrl>(const QJsonValue& value, const QString& what)
|
|
||||||
{
|
{
|
||||||
const QString string = ensureIsType<QString>(value, what);
|
const QString string = ensureIsType<QString>(value, what);
|
||||||
if (string.isEmpty()) {
|
if (string.isEmpty())
|
||||||
|
{
|
||||||
return QUrl();
|
return QUrl();
|
||||||
}
|
}
|
||||||
const QUrl url = QUrl(string, QUrl::StrictMode);
|
const QUrl url = QUrl(string, QUrl::StrictMode);
|
||||||
if (!url.isValid()) {
|
if (!url.isValid())
|
||||||
|
{
|
||||||
throw JsonException(what + " is not a correctly formatted URL");
|
throw JsonException(what + " is not a correctly formatted URL");
|
||||||
}
|
}
|
||||||
return url;
|
return url;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template<> QDir requireIsType<QDir>(const QJsonValue &value, const QString &what)
|
||||||
QDir requireIsType<QDir>(const QJsonValue& value, const QString& what)
|
|
||||||
{
|
{
|
||||||
const QString string = requireIsType<QString>(value, what);
|
const QString string = requireIsType<QString>(value, what);
|
||||||
// FIXME: does not handle invalid characters!
|
// FIXME: does not handle invalid characters!
|
||||||
return QDir::current().absoluteFilePath(string);
|
return QDir::current().absoluteFilePath(string);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template<> QUuid requireIsType<QUuid>(const QJsonValue &value, const QString &what)
|
||||||
QUuid requireIsType<QUuid>(const QJsonValue& value, const QString& what)
|
|
||||||
{
|
{
|
||||||
const QString string = requireIsType<QString>(value, what);
|
const QString string = requireIsType<QString>(value, what);
|
||||||
const QUuid uuid = QUuid(string);
|
const QUuid uuid = QUuid(string);
|
||||||
if (uuid.toString() != string) // converts back => valid
|
if (uuid.toString() != string) // converts back => valid
|
||||||
{
|
{
|
||||||
throw JsonException(what + " is not a valid UUID");
|
throw JsonException(what + " is not a valid UUID");
|
||||||
}
|
}
|
||||||
return uuid;
|
return uuid;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template<> QJsonObject requireIsType<QJsonObject>(const QJsonValue &value, const QString &what)
|
||||||
QJsonObject requireIsType<QJsonObject>(const QJsonValue& value, const QString& what)
|
|
||||||
{
|
{
|
||||||
if (!value.isObject()) {
|
if (!value.isObject())
|
||||||
|
{
|
||||||
throw JsonException(what + " is not an object");
|
throw JsonException(what + " is not an object");
|
||||||
}
|
}
|
||||||
return value.toObject();
|
return value.toObject();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template<> QVariant requireIsType<QVariant>(const QJsonValue &value, const QString &what)
|
||||||
QVariant requireIsType<QVariant>(const QJsonValue& value, const QString& what)
|
|
||||||
{
|
{
|
||||||
if (value.isNull() || value.isUndefined()) {
|
if (value.isNull() || value.isUndefined())
|
||||||
|
{
|
||||||
throw JsonException(what + " is null or undefined");
|
throw JsonException(what + " is null or undefined");
|
||||||
}
|
}
|
||||||
return value.toVariant();
|
return value.toVariant();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <>
|
template<> QJsonValue requireIsType<QJsonValue>(const QJsonValue &value, const QString &what)
|
||||||
QJsonValue requireIsType<QJsonValue>(const QJsonValue& value, const QString& what)
|
|
||||||
{
|
{
|
||||||
if (value.isNull() || value.isUndefined()) {
|
if (value.isNull() || value.isUndefined())
|
||||||
|
{
|
||||||
throw JsonException(what + " is null or undefined");
|
throw JsonException(what + " is null or undefined");
|
||||||
}
|
}
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace Json
|
}
|
||||||
|
202
launcher/Json.h
202
launcher/Json.h
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
@ -35,71 +35,74 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QDateTime>
|
|
||||||
#include <QDir>
|
|
||||||
#include <QJsonArray>
|
|
||||||
#include <QJsonDocument>
|
#include <QJsonDocument>
|
||||||
|
#include <QJsonArray>
|
||||||
#include <QJsonObject>
|
#include <QJsonObject>
|
||||||
|
#include <QDateTime>
|
||||||
#include <QUrl>
|
#include <QUrl>
|
||||||
|
#include <QDir>
|
||||||
#include <QUuid>
|
#include <QUuid>
|
||||||
#include <QVariant>
|
#include <QVariant>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
#include "Exception.h"
|
#include "Exception.h"
|
||||||
|
|
||||||
namespace Json {
|
namespace Json
|
||||||
class JsonException : public ::Exception {
|
{
|
||||||
public:
|
class JsonException : public ::Exception
|
||||||
JsonException(const QString& message) : Exception(message) {}
|
{
|
||||||
|
public:
|
||||||
|
JsonException(const QString &message) : Exception(message) {}
|
||||||
};
|
};
|
||||||
|
|
||||||
/// @throw FileSystemException
|
/// @throw FileSystemException
|
||||||
void write(const QJsonDocument& doc, const QString& filename);
|
void write(const QJsonDocument &doc, const QString &filename);
|
||||||
/// @throw FileSystemException
|
/// @throw FileSystemException
|
||||||
void write(const QJsonObject& object, const QString& filename);
|
void write(const QJsonObject &object, const QString &filename);
|
||||||
/// @throw FileSystemException
|
/// @throw FileSystemException
|
||||||
void write(const QJsonArray& array, const QString& filename);
|
void write(const QJsonArray &array, const QString &filename);
|
||||||
|
|
||||||
QByteArray toText(const QJsonObject& obj);
|
QByteArray toText(const QJsonObject &obj);
|
||||||
QByteArray toText(const QJsonArray& array);
|
QByteArray toText(const QJsonArray &array);
|
||||||
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
QJsonDocument requireDocument(const QByteArray& data, const QString& what = "Document");
|
QJsonDocument requireDocument(const QByteArray &data, const QString &what = "Document");
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
QJsonDocument requireDocument(const QString& filename, const QString& what = "Document");
|
QJsonDocument requireDocument(const QString &filename, const QString &what = "Document");
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
QJsonObject requireObject(const QJsonDocument& doc, const QString& what = "Document");
|
QJsonObject requireObject(const QJsonDocument &doc, const QString &what = "Document");
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
QJsonArray requireArray(const QJsonDocument& doc, const QString& what = "Document");
|
QJsonArray requireArray(const QJsonDocument &doc, const QString &what = "Document");
|
||||||
|
|
||||||
/////////////////// WRITING ////////////////////
|
/////////////////// WRITING ////////////////////
|
||||||
|
|
||||||
void writeString(QJsonObject& to, const QString& key, const QString& value);
|
void writeString(QJsonObject & to, const QString &key, const QString &value);
|
||||||
void writeStringList(QJsonObject& to, const QString& key, const QStringList& values);
|
void writeStringList(QJsonObject & to, const QString &key, const QStringList &values);
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
QJsonValue toJson(const T& t)
|
QJsonValue toJson(const T &t)
|
||||||
{
|
{
|
||||||
return QJsonValue(t);
|
return QJsonValue(t);
|
||||||
}
|
}
|
||||||
template <>
|
template<>
|
||||||
QJsonValue toJson<QUrl>(const QUrl& url);
|
QJsonValue toJson<QUrl>(const QUrl &url);
|
||||||
template <>
|
template<>
|
||||||
QJsonValue toJson<QByteArray>(const QByteArray& data);
|
QJsonValue toJson<QByteArray>(const QByteArray &data);
|
||||||
template <>
|
template<>
|
||||||
QJsonValue toJson<QDateTime>(const QDateTime& datetime);
|
QJsonValue toJson<QDateTime>(const QDateTime &datetime);
|
||||||
template <>
|
template<>
|
||||||
QJsonValue toJson<QDir>(const QDir& dir);
|
QJsonValue toJson<QDir>(const QDir &dir);
|
||||||
template <>
|
template<>
|
||||||
QJsonValue toJson<QUuid>(const QUuid& uuid);
|
QJsonValue toJson<QUuid>(const QUuid &uuid);
|
||||||
template <>
|
template<>
|
||||||
QJsonValue toJson<QVariant>(const QVariant& variant);
|
QJsonValue toJson<QVariant>(const QVariant &variant);
|
||||||
|
|
||||||
template <typename T>
|
template<typename T>
|
||||||
QJsonArray toJsonArray(const QList<T>& container)
|
QJsonArray toJsonArray(const QList<T> &container)
|
||||||
{
|
{
|
||||||
QJsonArray array;
|
QJsonArray array;
|
||||||
for (const T item : container) {
|
for (const T item : container)
|
||||||
|
{
|
||||||
array.append(toJson<T>(item));
|
array.append(toJson<T>(item));
|
||||||
}
|
}
|
||||||
return array;
|
return array;
|
||||||
@ -109,110 +112,106 @@ QJsonArray toJsonArray(const QList<T>& container)
|
|||||||
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <typename T>
|
template <typename T>
|
||||||
T requireIsType(const QJsonValue& value, const QString& what = "Value");
|
T requireIsType(const QJsonValue &value, const QString &what = "Value");
|
||||||
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <>
|
template<> double requireIsType<double>(const QJsonValue &value, const QString &what);
|
||||||
double requireIsType<double>(const QJsonValue& value, const QString& what);
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <>
|
template<> bool requireIsType<bool>(const QJsonValue &value, const QString &what);
|
||||||
bool requireIsType<bool>(const QJsonValue& value, const QString& what);
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <>
|
template<> int requireIsType<int>(const QJsonValue &value, const QString &what);
|
||||||
int requireIsType<int>(const QJsonValue& value, const QString& what);
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <>
|
template<> QJsonObject requireIsType<QJsonObject>(const QJsonValue &value, const QString &what);
|
||||||
QJsonObject requireIsType<QJsonObject>(const QJsonValue& value, const QString& what);
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <>
|
template<> QJsonArray requireIsType<QJsonArray>(const QJsonValue &value, const QString &what);
|
||||||
QJsonArray requireIsType<QJsonArray>(const QJsonValue& value, const QString& what);
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <>
|
template<> QJsonValue requireIsType<QJsonValue>(const QJsonValue &value, const QString &what);
|
||||||
QJsonValue requireIsType<QJsonValue>(const QJsonValue& value, const QString& what);
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <>
|
template<> QByteArray requireIsType<QByteArray>(const QJsonValue &value, const QString &what);
|
||||||
QByteArray requireIsType<QByteArray>(const QJsonValue& value, const QString& what);
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <>
|
template<> QDateTime requireIsType<QDateTime>(const QJsonValue &value, const QString &what);
|
||||||
QDateTime requireIsType<QDateTime>(const QJsonValue& value, const QString& what);
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <>
|
template<> QVariant requireIsType<QVariant>(const QJsonValue &value, const QString &what);
|
||||||
QVariant requireIsType<QVariant>(const QJsonValue& value, const QString& what);
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <>
|
template<> QString requireIsType<QString>(const QJsonValue &value, const QString &what);
|
||||||
QString requireIsType<QString>(const QJsonValue& value, const QString& what);
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <>
|
template<> QUuid requireIsType<QUuid>(const QJsonValue &value, const QString &what);
|
||||||
QUuid requireIsType<QUuid>(const QJsonValue& value, const QString& what);
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <>
|
template<> QDir requireIsType<QDir>(const QJsonValue &value, const QString &what);
|
||||||
QDir requireIsType<QDir>(const QJsonValue& value, const QString& what);
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <>
|
template<> QUrl requireIsType<QUrl>(const QJsonValue &value, const QString &what);
|
||||||
QUrl requireIsType<QUrl>(const QJsonValue& value, const QString& what);
|
|
||||||
|
|
||||||
// the following functions are higher level functions, that make use of the above functions for
|
// the following functions are higher level functions, that make use of the above functions for
|
||||||
// type conversion
|
// type conversion
|
||||||
template <typename T>
|
template <typename T>
|
||||||
T ensureIsType(const QJsonValue& value, const T default_ = T(), const QString& what = "Value")
|
T ensureIsType(const QJsonValue &value, const T default_ = T(), const QString &what = "Value")
|
||||||
{
|
{
|
||||||
if (value.isUndefined() || value.isNull()) {
|
if (value.isUndefined() || value.isNull())
|
||||||
|
{
|
||||||
return default_;
|
return default_;
|
||||||
}
|
}
|
||||||
try {
|
try
|
||||||
|
{
|
||||||
return requireIsType<T>(value, what);
|
return requireIsType<T>(value, what);
|
||||||
} catch (const JsonException&) {
|
}
|
||||||
|
catch (const JsonException &)
|
||||||
|
{
|
||||||
return default_;
|
return default_;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <typename T>
|
template <typename T>
|
||||||
T requireIsType(const QJsonObject& parent, const QString& key, const QString& what = "__placeholder__")
|
T requireIsType(const QJsonObject &parent, const QString &key, const QString &what = "__placeholder__")
|
||||||
{
|
{
|
||||||
const QString localWhat = QString(what).replace("__placeholder__", '\'' + key + '\'');
|
const QString localWhat = QString(what).replace("__placeholder__", '\'' + key + '\'');
|
||||||
if (!parent.contains(key)) {
|
if (!parent.contains(key))
|
||||||
|
{
|
||||||
throw JsonException(localWhat + "s parent does not contain " + localWhat);
|
throw JsonException(localWhat + "s parent does not contain " + localWhat);
|
||||||
}
|
}
|
||||||
return requireIsType<T>(parent.value(key), localWhat);
|
return requireIsType<T>(parent.value(key), localWhat);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
T ensureIsType(const QJsonObject& parent, const QString& key, const T default_ = T(), const QString& what = "__placeholder__")
|
T ensureIsType(const QJsonObject &parent, const QString &key, const T default_ = T(), const QString &what = "__placeholder__")
|
||||||
{
|
{
|
||||||
const QString localWhat = QString(what).replace("__placeholder__", '\'' + key + '\'');
|
const QString localWhat = QString(what).replace("__placeholder__", '\'' + key + '\'');
|
||||||
if (!parent.contains(key)) {
|
if (!parent.contains(key))
|
||||||
|
{
|
||||||
return default_;
|
return default_;
|
||||||
}
|
}
|
||||||
return ensureIsType<T>(parent.value(key), default_, localWhat);
|
return ensureIsType<T>(parent.value(key), default_, localWhat);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
QVector<T> requireIsArrayOf(const QJsonDocument& doc)
|
QVector<T> requireIsArrayOf(const QJsonDocument &doc)
|
||||||
{
|
{
|
||||||
const QJsonArray array = requireArray(doc);
|
const QJsonArray array = requireArray(doc);
|
||||||
QVector<T> out;
|
QVector<T> out;
|
||||||
for (const QJsonValue val : array) {
|
for (const QJsonValue val : array)
|
||||||
|
{
|
||||||
out.append(requireIsType<T>(val, "Document"));
|
out.append(requireIsType<T>(val, "Document"));
|
||||||
}
|
}
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
QVector<T> ensureIsArrayOf(const QJsonValue& value, const QString& what = "Value")
|
QVector<T> ensureIsArrayOf(const QJsonValue &value, const QString &what = "Value")
|
||||||
{
|
{
|
||||||
const QJsonArray array = ensureIsType<QJsonArray>(value, QJsonArray(), what);
|
const QJsonArray array = ensureIsType<QJsonArray>(value, QJsonArray(), what);
|
||||||
QVector<T> out;
|
QVector<T> out;
|
||||||
for (const QJsonValue val : array) {
|
for (const QJsonValue val : array)
|
||||||
|
{
|
||||||
out.append(requireIsType<T>(val, what));
|
out.append(requireIsType<T>(val, what));
|
||||||
}
|
}
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
QVector<T> ensureIsArrayOf(const QJsonValue& value, const QVector<T> default_, const QString& what = "Value")
|
QVector<T> ensureIsArrayOf(const QJsonValue &value, const QVector<T> default_, const QString &what = "Value")
|
||||||
{
|
{
|
||||||
if (value.isUndefined()) {
|
if (value.isUndefined())
|
||||||
|
{
|
||||||
return default_;
|
return default_;
|
||||||
}
|
}
|
||||||
return ensureIsArrayOf<T>(value, what);
|
return ensureIsArrayOf<T>(value, what);
|
||||||
@ -220,46 +219,45 @@ QVector<T> ensureIsArrayOf(const QJsonValue& value, const QVector<T> default_, c
|
|||||||
|
|
||||||
/// @throw JsonException
|
/// @throw JsonException
|
||||||
template <typename T>
|
template <typename T>
|
||||||
QVector<T> requireIsArrayOf(const QJsonObject& parent, const QString& key, const QString& what = "__placeholder__")
|
QVector<T> requireIsArrayOf(const QJsonObject &parent, const QString &key, const QString &what = "__placeholder__")
|
||||||
{
|
{
|
||||||
const QString localWhat = QString(what).replace("__placeholder__", '\'' + key + '\'');
|
const QString localWhat = QString(what).replace("__placeholder__", '\'' + key + '\'');
|
||||||
if (!parent.contains(key)) {
|
if (!parent.contains(key))
|
||||||
|
{
|
||||||
throw JsonException(localWhat + "s parent does not contain " + localWhat);
|
throw JsonException(localWhat + "s parent does not contain " + localWhat);
|
||||||
}
|
}
|
||||||
return ensureIsArrayOf<T>(parent.value(key), localWhat);
|
return ensureIsArrayOf<T>(parent.value(key), localWhat);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
QVector<T> ensureIsArrayOf(const QJsonObject& parent,
|
QVector<T> ensureIsArrayOf(const QJsonObject &parent, const QString &key,
|
||||||
const QString& key,
|
const QVector<T> &default_ = QVector<T>(), const QString &what = "__placeholder__")
|
||||||
const QVector<T>& default_ = QVector<T>(),
|
|
||||||
const QString& what = "__placeholder__")
|
|
||||||
{
|
{
|
||||||
const QString localWhat = QString(what).replace("__placeholder__", '\'' + key + '\'');
|
const QString localWhat = QString(what).replace("__placeholder__", '\'' + key + '\'');
|
||||||
if (!parent.contains(key)) {
|
if (!parent.contains(key))
|
||||||
|
{
|
||||||
return default_;
|
return default_;
|
||||||
}
|
}
|
||||||
return ensureIsArrayOf<T>(parent.value(key), default_, localWhat);
|
return ensureIsArrayOf<T>(parent.value(key), default_, localWhat);
|
||||||
}
|
}
|
||||||
|
|
||||||
// this macro part could be replaced by variadic functions that just pass on their arguments, but that wouldn't work well with IDE helpers
|
// this macro part could be replaced by variadic functions that just pass on their arguments, but that wouldn't work well with IDE helpers
|
||||||
#define JSON_HELPERFUNCTIONS(NAME, TYPE) \
|
#define JSON_HELPERFUNCTIONS(NAME, TYPE) \
|
||||||
inline TYPE require##NAME(const QJsonValue& value, const QString& what = "Value") \
|
inline TYPE require##NAME(const QJsonValue &value, const QString &what = "Value") \
|
||||||
{ \
|
{ \
|
||||||
return requireIsType<TYPE>(value, what); \
|
return requireIsType<TYPE>(value, what); \
|
||||||
} \
|
} \
|
||||||
inline TYPE ensure##NAME(const QJsonValue& value, const TYPE default_ = TYPE(), const QString& what = "Value") \
|
inline TYPE ensure##NAME(const QJsonValue &value, const TYPE default_ = TYPE(), const QString &what = "Value") \
|
||||||
{ \
|
{ \
|
||||||
return ensureIsType<TYPE>(value, default_, what); \
|
return ensureIsType<TYPE>(value, default_, what); \
|
||||||
} \
|
} \
|
||||||
inline TYPE require##NAME(const QJsonObject& parent, const QString& key, const QString& what = "__placeholder__") \
|
inline TYPE require##NAME(const QJsonObject &parent, const QString &key, const QString &what = "__placeholder__") \
|
||||||
{ \
|
{ \
|
||||||
return requireIsType<TYPE>(parent, key, what); \
|
return requireIsType<TYPE>(parent, key, what); \
|
||||||
} \
|
} \
|
||||||
inline TYPE ensure##NAME(const QJsonObject& parent, const QString& key, const TYPE default_ = TYPE(), \
|
inline TYPE ensure##NAME(const QJsonObject &parent, const QString &key, const TYPE default_ = TYPE(), const QString &what = "__placeholder") \
|
||||||
const QString& what = "__placeholder") \
|
{ \
|
||||||
{ \
|
return ensureIsType<TYPE>(parent, key, default_, what); \
|
||||||
return ensureIsType<TYPE>(parent, key, default_, what); \
|
|
||||||
}
|
}
|
||||||
|
|
||||||
JSON_HELPERFUNCTIONS(Array, QJsonArray)
|
JSON_HELPERFUNCTIONS(Array, QJsonArray)
|
||||||
@ -278,5 +276,5 @@ JSON_HELPERFUNCTIONS(Variant, QVariant)
|
|||||||
|
|
||||||
#undef JSON_HELPERFUNCTIONS
|
#undef JSON_HELPERFUNCTIONS
|
||||||
|
|
||||||
} // namespace Json
|
}
|
||||||
using JSONValidationError = Json::JsonException;
|
using JSONValidationError = Json::JsonException;
|
||||||
|
@ -1,26 +1,42 @@
|
|||||||
#include "KonamiCode.h"
|
#include "KonamiCode.h"
|
||||||
|
|
||||||
#include <QDebug>
|
|
||||||
#include <array>
|
#include <array>
|
||||||
|
#include <QDebug>
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
const std::array<Qt::Key, 10> konamiCode = { { Qt::Key_Up, Qt::Key_Up, Qt::Key_Down, Qt::Key_Down, Qt::Key_Left, Qt::Key_Right,
|
const std::array<Qt::Key, 10> konamiCode =
|
||||||
Qt::Key_Left, Qt::Key_Right, Qt::Key_B, Qt::Key_A } };
|
{
|
||||||
|
{
|
||||||
|
Qt::Key_Up, Qt::Key_Up,
|
||||||
|
Qt::Key_Down, Qt::Key_Down,
|
||||||
|
Qt::Key_Left, Qt::Key_Right,
|
||||||
|
Qt::Key_Left, Qt::Key_Right,
|
||||||
|
Qt::Key_B, Qt::Key_A
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
KonamiCode::KonamiCode(QObject* parent) : QObject(parent)
|
||||||
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
KonamiCode::KonamiCode(QObject* parent) : QObject(parent) {}
|
|
||||||
|
|
||||||
void KonamiCode::input(QEvent* event)
|
void KonamiCode::input(QEvent* event)
|
||||||
{
|
{
|
||||||
if (event->type() == QEvent::KeyPress) {
|
if( event->type() == QEvent::KeyPress )
|
||||||
QKeyEvent* keyEvent = static_cast<QKeyEvent*>(event);
|
{
|
||||||
|
QKeyEvent *keyEvent = static_cast<QKeyEvent*>( event );
|
||||||
auto key = Qt::Key(keyEvent->key());
|
auto key = Qt::Key(keyEvent->key());
|
||||||
if (key == konamiCode[m_progress]) {
|
if(key == konamiCode[m_progress])
|
||||||
m_progress++;
|
{
|
||||||
} else {
|
m_progress ++;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
m_progress = 0;
|
m_progress = 0;
|
||||||
}
|
}
|
||||||
if (m_progress == static_cast<int>(konamiCode.size())) {
|
if(m_progress == static_cast<int>(konamiCode.size()))
|
||||||
|
{
|
||||||
m_progress = 0;
|
m_progress = 0;
|
||||||
emit triggered();
|
emit triggered();
|
||||||
}
|
}
|
||||||
|
@ -2,15 +2,16 @@
|
|||||||
|
|
||||||
#include <QKeyEvent>
|
#include <QKeyEvent>
|
||||||
|
|
||||||
class KonamiCode : public QObject {
|
class KonamiCode : public QObject
|
||||||
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
KonamiCode(QObject* parent = 0);
|
KonamiCode(QObject *parent = 0);
|
||||||
void input(QEvent* event);
|
void input(QEvent *event);
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void triggered();
|
void triggered();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
int m_progress = 0;
|
int m_progress = 0;
|
||||||
};
|
};
|
||||||
|
@ -1,8 +1,7 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
* Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me>
|
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -35,41 +34,44 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "LaunchController.h"
|
#include "LaunchController.h"
|
||||||
#include "Application.h"
|
|
||||||
#include "minecraft/auth/AccountList.h"
|
#include "minecraft/auth/AccountList.h"
|
||||||
|
#include "Application.h"
|
||||||
|
|
||||||
#include "ui/InstanceWindow.h"
|
|
||||||
#include "ui/MainWindow.h"
|
#include "ui/MainWindow.h"
|
||||||
|
#include "ui/InstanceWindow.h"
|
||||||
#include "ui/dialogs/CustomMessageBox.h"
|
#include "ui/dialogs/CustomMessageBox.h"
|
||||||
#include "ui/dialogs/EditAccountDialog.h"
|
|
||||||
#include "ui/dialogs/ProfileSelectDialog.h"
|
#include "ui/dialogs/ProfileSelectDialog.h"
|
||||||
#include "ui/dialogs/ProfileSetupDialog.h"
|
|
||||||
#include "ui/dialogs/ProgressDialog.h"
|
#include "ui/dialogs/ProgressDialog.h"
|
||||||
|
#include "ui/dialogs/EditAccountDialog.h"
|
||||||
|
#include "ui/dialogs/ProfileSetupDialog.h"
|
||||||
|
|
||||||
#include <QHostAddress>
|
|
||||||
#include <QHostInfo>
|
|
||||||
#include <QInputDialog>
|
|
||||||
#include <QLineEdit>
|
#include <QLineEdit>
|
||||||
#include <QList>
|
#include <QInputDialog>
|
||||||
#include <QPushButton>
|
|
||||||
#include <QStringList>
|
#include <QStringList>
|
||||||
|
#include <QHostInfo>
|
||||||
|
#include <QList>
|
||||||
|
#include <QHostAddress>
|
||||||
|
#include <QPushButton>
|
||||||
|
|
||||||
#include "BuildConfig.h"
|
#include "BuildConfig.h"
|
||||||
#include "JavaCommon.h"
|
#include "JavaCommon.h"
|
||||||
#include "launch/steps/TextPrint.h"
|
|
||||||
#include "minecraft/auth/AccountTask.h"
|
|
||||||
#include "tasks/Task.h"
|
#include "tasks/Task.h"
|
||||||
|
#include "minecraft/auth/AccountTask.h"
|
||||||
|
#include "launch/steps/TextPrint.h"
|
||||||
|
|
||||||
LaunchController::LaunchController(QObject* parent) : Task(parent) {}
|
LaunchController::LaunchController(QObject *parent) : Task(parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
void LaunchController::executeTask()
|
void LaunchController::executeTask()
|
||||||
{
|
{
|
||||||
if (!m_instance) {
|
if (!m_instance)
|
||||||
|
{
|
||||||
emitFailed(tr("No instance specified!"));
|
emitFailed(tr("No instance specified!"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!JavaCommon::checkJVMArgs(m_instance->settings()->get("JvmArgs").toString(), m_parentWidget)) {
|
if(!JavaCommon::checkJVMArgs(m_instance->settings()->get("JvmArgs").toString(), m_parentWidget)) {
|
||||||
emitFailed(tr("Invalid Java arguments specified. Please fix this first."));
|
emitFailed(tr("Invalid Java arguments specified. Please fix this first."));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -79,25 +81,32 @@ void LaunchController::executeTask()
|
|||||||
|
|
||||||
void LaunchController::decideAccount()
|
void LaunchController::decideAccount()
|
||||||
{
|
{
|
||||||
if (m_accountToUse) {
|
if(m_accountToUse) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find an account to use.
|
// Find an account to use.
|
||||||
auto accounts = APPLICATION->accounts();
|
auto accounts = APPLICATION->accounts();
|
||||||
if (accounts->count() <= 0) {
|
if (accounts->count() <= 0)
|
||||||
|
{
|
||||||
// Tell the user they need to log in at least one account in order to play.
|
// 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"),
|
auto reply = CustomMessageBox::selectable(
|
||||||
tr("In order to play Minecraft, you must have at least one Microsoft "
|
m_parentWidget,
|
||||||
"account which owns Minecraft logged in. "
|
tr("No Accounts"),
|
||||||
"Would you like to open the account manager to add an account now?"),
|
tr("In order to play Minecraft, you must have at least one Microsoft or Mojang "
|
||||||
QMessageBox::Information, QMessageBox::Yes | QMessageBox::No)
|
"account logged in. Mojang accounts can only be used offline. "
|
||||||
->exec();
|
"Would you like to open the account manager to add an account now?"),
|
||||||
|
QMessageBox::Information,
|
||||||
|
QMessageBox::Yes | QMessageBox::No
|
||||||
|
)->exec();
|
||||||
|
|
||||||
if (reply == QMessageBox::Yes) {
|
if (reply == QMessageBox::Yes)
|
||||||
|
{
|
||||||
// Open the account manager.
|
// Open the account manager.
|
||||||
APPLICATION->ShowGlobalSettings(m_parentWidget, "accounts");
|
APPLICATION->ShowGlobalSettings(m_parentWidget, "accounts");
|
||||||
} else if (reply == QMessageBox::No) {
|
}
|
||||||
|
else if (reply == QMessageBox::No)
|
||||||
|
{
|
||||||
// Do not open "profile select" dialog.
|
// Do not open "profile select" dialog.
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -106,16 +115,20 @@ 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
|
// 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 instanceAccountId = m_instance->settings()->get("InstanceAccountId").toString();
|
||||||
auto instanceAccountIndex = accounts->findAccountByProfileId(instanceAccountId);
|
auto instanceAccountIndex = accounts->findAccountByProfileId(instanceAccountId);
|
||||||
if (instanceAccountIndex == -1 || instanceAccountId.isEmpty()) {
|
if (instanceAccountIndex == -1) {
|
||||||
m_accountToUse = accounts->defaultAccount();
|
m_accountToUse = accounts->defaultAccount();
|
||||||
} else {
|
} else {
|
||||||
m_accountToUse = accounts->at(instanceAccountIndex);
|
m_accountToUse = accounts->at(instanceAccountIndex);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!m_accountToUse) {
|
if (!m_accountToUse)
|
||||||
|
{
|
||||||
// If no default account is set, ask the user which one to use.
|
// If no default account is set, ask the user which one to use.
|
||||||
ProfileSelectDialog selectDialog(tr("Which account would you like to use?"), ProfileSelectDialog::GlobalDefaultCheckbox,
|
ProfileSelectDialog selectDialog(
|
||||||
m_parentWidget);
|
tr("Which account would you like to use?"),
|
||||||
|
ProfileSelectDialog::GlobalDefaultCheckbox,
|
||||||
|
m_parentWidget
|
||||||
|
);
|
||||||
|
|
||||||
selectDialog.exec();
|
selectDialog.exec();
|
||||||
|
|
||||||
@ -129,12 +142,13 @@ void LaunchController::decideAccount()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void LaunchController::login()
|
|
||||||
{
|
void LaunchController::login() {
|
||||||
decideAccount();
|
decideAccount();
|
||||||
|
|
||||||
// if no account is selected, we bail
|
// if no account is selected, we bail
|
||||||
if (!m_accountToUse) {
|
if (!m_accountToUse)
|
||||||
|
{
|
||||||
emitFailed(tr("No account selected for launch."));
|
emitFailed(tr("No account selected for launch."));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -143,11 +157,15 @@ void LaunchController::login()
|
|||||||
bool tryagain = true;
|
bool tryagain = true;
|
||||||
unsigned int tries = 0;
|
unsigned int tries = 0;
|
||||||
|
|
||||||
while (tryagain) {
|
while (tryagain)
|
||||||
|
{
|
||||||
if (tries > 0 && tries % 3 == 0) {
|
if (tries > 0 && tries % 3 == 0) {
|
||||||
auto result =
|
auto result = QMessageBox::question(
|
||||||
QMessageBox::question(m_parentWidget, tr("Continue launch?"),
|
m_parentWidget,
|
||||||
tr("It looks like we couldn't launch after %1 tries. Do you want to continue trying?").arg(tries));
|
tr("Continue launch?"),
|
||||||
|
tr("It looks like we couldn't launch after %1 tries. Do you want to continue trying?")
|
||||||
|
.arg(tries)
|
||||||
|
);
|
||||||
|
|
||||||
if (result == QMessageBox::No) {
|
if (result == QMessageBox::No) {
|
||||||
emitAborted();
|
emitAborted();
|
||||||
@ -161,48 +179,60 @@ void LaunchController::login()
|
|||||||
m_accountToUse->fillSession(m_session);
|
m_accountToUse->fillSession(m_session);
|
||||||
|
|
||||||
// Launch immediately in true offline mode
|
// Launch immediately in true offline mode
|
||||||
if (m_accountToUse->isOffline()) {
|
if(m_accountToUse->isOffline()) {
|
||||||
launchInstance();
|
launchInstance();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (m_accountToUse->accountState()) {
|
switch(m_accountToUse->accountState()) {
|
||||||
case AccountState::Offline: {
|
case AccountState::Offline: {
|
||||||
m_session->wants_online = false;
|
m_session->wants_online = false;
|
||||||
}
|
}
|
||||||
/* fallthrough */
|
/* fallthrough */
|
||||||
case AccountState::Online: {
|
case AccountState::Online: {
|
||||||
if (!m_session->wants_online) {
|
if(!m_session->wants_online) {
|
||||||
// we ask the user for a player name
|
// we ask the user for a player name
|
||||||
bool ok = false;
|
bool ok = false;
|
||||||
|
|
||||||
QString message = tr("Choose your offline mode player name.");
|
QString message = tr("Choose your offline mode player name.");
|
||||||
if (m_session->demo) {
|
if(m_session->demo) {
|
||||||
message = tr("Choose your demo mode player name.");
|
message = tr("Choose your demo mode player name.");
|
||||||
}
|
}
|
||||||
|
|
||||||
QString lastOfflinePlayerName = APPLICATION->settings()->get("LastOfflinePlayerName").toString();
|
QString lastOfflinePlayerName = APPLICATION->settings()->get("LastOfflinePlayerName").toString();
|
||||||
QString usedname = lastOfflinePlayerName.isEmpty() ? m_session->player_name : lastOfflinePlayerName;
|
QString usedname = lastOfflinePlayerName.isEmpty() ? m_session->player_name : lastOfflinePlayerName;
|
||||||
QString name = QInputDialog::getText(m_parentWidget, tr("Player name"), message, QLineEdit::Normal, usedname, &ok);
|
QString name = QInputDialog::getText(
|
||||||
if (!ok) {
|
m_parentWidget,
|
||||||
|
tr("Player name"),
|
||||||
|
message,
|
||||||
|
QLineEdit::Normal,
|
||||||
|
usedname,
|
||||||
|
&ok
|
||||||
|
);
|
||||||
|
if (!ok)
|
||||||
|
{
|
||||||
tryagain = false;
|
tryagain = false;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (name.length()) {
|
if (name.length())
|
||||||
|
{
|
||||||
usedname = name;
|
usedname = name;
|
||||||
APPLICATION->settings()->set("LastOfflinePlayerName", usedname);
|
APPLICATION->settings()->set("LastOfflinePlayerName", usedname);
|
||||||
}
|
}
|
||||||
m_session->MakeOffline(usedname);
|
m_session->MakeOffline(usedname);
|
||||||
// offline flavored game from here :3
|
// offline flavored game from here :3
|
||||||
}
|
}
|
||||||
if (m_accountToUse->ownsMinecraft()) {
|
if(m_accountToUse->ownsMinecraft()) {
|
||||||
if (!m_accountToUse->hasProfile()) {
|
if(!m_accountToUse->hasProfile()) {
|
||||||
// Now handle setting up a profile name here...
|
// Now handle setting up a profile name here...
|
||||||
ProfileSetupDialog dialog(m_accountToUse, m_parentWidget);
|
ProfileSetupDialog dialog(m_accountToUse, m_parentWidget);
|
||||||
if (dialog.exec() == QDialog::Accepted) {
|
if (dialog.exec() == QDialog::Accepted)
|
||||||
|
{
|
||||||
tryagain = true;
|
tryagain = true;
|
||||||
continue;
|
continue;
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
emitFailed(tr("Received undetermined session status during login."));
|
emitFailed(tr("Received undetermined session status during login."));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -210,24 +240,24 @@ void LaunchController::login()
|
|||||||
// we own Minecraft, there is a profile, it's all ready to go!
|
// we own Minecraft, there is a profile, it's all ready to go!
|
||||||
launchInstance();
|
launchInstance();
|
||||||
return;
|
return;
|
||||||
} else {
|
}
|
||||||
|
else {
|
||||||
// play demo ?
|
// play demo ?
|
||||||
QMessageBox box(m_parentWidget);
|
QMessageBox box(m_parentWidget);
|
||||||
box.setWindowTitle(tr("Play demo?"));
|
box.setWindowTitle(tr("Play demo?"));
|
||||||
box.setText(
|
box.setText(tr("This account does not own Minecraft.\nYou need to purchase the game first to play it.\n\nDo you want to play the demo?"));
|
||||||
tr("This account does not own Minecraft.\nYou need to purchase the game first to play it.\n\nDo you want to play "
|
|
||||||
"the demo?"));
|
|
||||||
box.setIcon(QMessageBox::Warning);
|
box.setIcon(QMessageBox::Warning);
|
||||||
auto demoButton = box.addButton(tr("Play Demo"), QMessageBox::ButtonRole::YesRole);
|
auto demoButton = box.addButton(tr("Play Demo"), QMessageBox::ButtonRole::YesRole);
|
||||||
auto cancelButton = box.addButton(tr("Cancel"), QMessageBox::ButtonRole::NoRole);
|
auto cancelButton = box.addButton(tr("Cancel"), QMessageBox::ButtonRole::NoRole);
|
||||||
box.setDefaultButton(cancelButton);
|
box.setDefaultButton(cancelButton);
|
||||||
|
|
||||||
box.exec();
|
box.exec();
|
||||||
if (box.clickedButton() == demoButton) {
|
if(box.clickedButton() == demoButton) {
|
||||||
// play demo here
|
// play demo here
|
||||||
m_session->MakeDemo();
|
m_session->MakeDemo();
|
||||||
launchInstance();
|
launchInstance();
|
||||||
} else {
|
}
|
||||||
|
else {
|
||||||
emitFailed(tr("Launch cancelled - account does not own Minecraft."));
|
emitFailed(tr("Launch cancelled - account does not own Minecraft."));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -242,7 +272,8 @@ void LaunchController::login()
|
|||||||
case AccountState::Working: {
|
case AccountState::Working: {
|
||||||
// refresh is in progress, we need to wait for it to finish to proceed.
|
// refresh is in progress, we need to wait for it to finish to proceed.
|
||||||
ProgressDialog progDialog(m_parentWidget);
|
ProgressDialog progDialog(m_parentWidget);
|
||||||
if (m_online) {
|
if (m_online)
|
||||||
|
{
|
||||||
progDialog.setSkipButton(true, tr("Play Offline"));
|
progDialog.setSkipButton(true, tr("Play Offline"));
|
||||||
}
|
}
|
||||||
auto task = m_accountToUse->currentTask();
|
auto task = m_accountToUse->currentTask();
|
||||||
@ -257,24 +288,37 @@ void LaunchController::login()
|
|||||||
*/
|
*/
|
||||||
case AccountState::Expired: {
|
case AccountState::Expired: {
|
||||||
auto errorString = tr("The account has expired and needs to be logged into manually again.");
|
auto errorString = tr("The account has expired and needs to be logged into manually again.");
|
||||||
QMessageBox::warning(m_parentWidget, tr("Account refresh failed"), errorString, QMessageBox::StandardButton::Ok,
|
QMessageBox::warning(
|
||||||
QMessageBox::StandardButton::Ok);
|
m_parentWidget,
|
||||||
|
tr("Account refresh failed"),
|
||||||
|
errorString,
|
||||||
|
QMessageBox::StandardButton::Ok,
|
||||||
|
QMessageBox::StandardButton::Ok
|
||||||
|
);
|
||||||
emitFailed(errorString);
|
emitFailed(errorString);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
case AccountState::Disabled: {
|
case AccountState::Disabled: {
|
||||||
auto errorString = tr("The launcher's client identification has changed. Please remove this account and add it again.");
|
auto errorString = tr("The launcher's client identification has changed. Please remove this account and add it again.");
|
||||||
QMessageBox::warning(m_parentWidget, tr("Client identification changed"), errorString, QMessageBox::StandardButton::Ok,
|
QMessageBox::warning(
|
||||||
QMessageBox::StandardButton::Ok);
|
m_parentWidget,
|
||||||
|
tr("Client identification changed"),
|
||||||
|
errorString,
|
||||||
|
QMessageBox::StandardButton::Ok,
|
||||||
|
QMessageBox::StandardButton::Ok
|
||||||
|
);
|
||||||
emitFailed(errorString);
|
emitFailed(errorString);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
case AccountState::Gone: {
|
case AccountState::Gone: {
|
||||||
auto errorString =
|
auto errorString = tr("The account no longer exists on the servers. It may have been migrated, in which case please add the new account you migrated this one to.");
|
||||||
tr("The account no longer exists on the servers. It may have been migrated, in which case please add the new account "
|
QMessageBox::warning(
|
||||||
"you migrated this one to.");
|
m_parentWidget,
|
||||||
QMessageBox::warning(m_parentWidget, tr("Account gone"), errorString, QMessageBox::StandardButton::Ok,
|
tr("Account gone"),
|
||||||
QMessageBox::StandardButton::Ok);
|
errorString,
|
||||||
|
QMessageBox::StandardButton::Ok,
|
||||||
|
QMessageBox::StandardButton::Ok
|
||||||
|
);
|
||||||
emitFailed(errorString);
|
emitFailed(errorString);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -288,45 +332,48 @@ void LaunchController::launchInstance()
|
|||||||
Q_ASSERT_X(m_instance != NULL, "launchInstance", "instance is NULL");
|
Q_ASSERT_X(m_instance != NULL, "launchInstance", "instance is NULL");
|
||||||
Q_ASSERT_X(m_session.get() != nullptr, "launchInstance", "session is NULL");
|
Q_ASSERT_X(m_session.get() != nullptr, "launchInstance", "session is NULL");
|
||||||
|
|
||||||
if (!m_instance->reloadSettings()) {
|
if(!m_instance->reloadSettings())
|
||||||
|
{
|
||||||
QMessageBox::critical(m_parentWidget, tr("Error!"), tr("Couldn't load the instance profile."));
|
QMessageBox::critical(m_parentWidget, tr("Error!"), tr("Couldn't load the instance profile."));
|
||||||
emitFailed(tr("Couldn't load the instance profile."));
|
emitFailed(tr("Couldn't load the instance profile."));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
m_launcher = m_instance->createLaunchTask(m_session, m_serverToJoin);
|
m_launcher = m_instance->createLaunchTask(m_session, m_serverToJoin);
|
||||||
if (!m_launcher) {
|
if (!m_launcher)
|
||||||
|
{
|
||||||
emitFailed(tr("Couldn't instantiate a launcher."));
|
emitFailed(tr("Couldn't instantiate a launcher."));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto console = qobject_cast<InstanceWindow*>(m_parentWidget);
|
auto console = qobject_cast<InstanceWindow *>(m_parentWidget);
|
||||||
auto showConsole = m_instance->settings()->get("ShowConsole").toBool();
|
auto showConsole = m_instance->settings()->get("ShowConsole").toBool();
|
||||||
if (!console && showConsole) {
|
if(!console && showConsole)
|
||||||
|
{
|
||||||
APPLICATION->showInstanceWindow(m_instance);
|
APPLICATION->showInstanceWindow(m_instance);
|
||||||
}
|
}
|
||||||
connect(m_launcher.get(), &LaunchTask::readyForLaunch, this, &LaunchController::readyForLaunch);
|
connect(m_launcher.get(), &LaunchTask::readyForLaunch, this, &LaunchController::readyForLaunch);
|
||||||
connect(m_launcher.get(), &LaunchTask::succeeded, this, &LaunchController::onSucceeded);
|
connect(m_launcher.get(), &LaunchTask::succeeded, this, &LaunchController::onSucceeded);
|
||||||
connect(m_launcher.get(), &LaunchTask::failed, this, &LaunchController::onFailed);
|
connect(m_launcher.get(), &LaunchTask::failed, this, &LaunchController::onFailed);
|
||||||
connect(m_launcher.get(), &LaunchTask::requestProgress, this, &LaunchController::onProgressRequested);
|
connect(m_launcher.get(), &LaunchTask::requestProgress, this, &LaunchController::onProgressRequested);
|
||||||
|
|
||||||
// Prepend Online and Auth Status
|
// Prepend Online and Auth Status
|
||||||
QString online_mode;
|
QString online_mode;
|
||||||
if (m_session->wants_online) {
|
if(m_session->wants_online) {
|
||||||
online_mode = "online";
|
online_mode = "online";
|
||||||
|
|
||||||
// Prepend Server Status
|
// Prepend Server Status
|
||||||
QStringList servers = { "authserver.mojang.com", "session.minecraft.net", "textures.minecraft.net", "api.mojang.com" };
|
QStringList servers = {"authserver.mojang.com", "session.minecraft.net", "textures.minecraft.net", "api.mojang.com"};
|
||||||
QString resolved_servers = "";
|
QString resolved_servers = "";
|
||||||
QHostInfo host_info;
|
QHostInfo host_info;
|
||||||
|
|
||||||
for (QString server : servers) {
|
for(QString server : servers) {
|
||||||
host_info = QHostInfo::fromName(server);
|
host_info = QHostInfo::fromName(server);
|
||||||
resolved_servers = resolved_servers + server + " resolves to:\n [";
|
resolved_servers = resolved_servers + server + " resolves to:\n [";
|
||||||
if (!host_info.addresses().isEmpty()) {
|
if(!host_info.addresses().isEmpty()) {
|
||||||
for (QHostAddress address : host_info.addresses()) {
|
for(QHostAddress address : host_info.addresses()) {
|
||||||
resolved_servers = resolved_servers + address.toString();
|
resolved_servers = resolved_servers + address.toString();
|
||||||
if (!host_info.addresses().endsWith(address)) {
|
if(!host_info.addresses().endsWith(address)) {
|
||||||
resolved_servers = resolved_servers + ", ";
|
resolved_servers = resolved_servers + ", ";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -340,13 +387,11 @@ void LaunchController::launchInstance()
|
|||||||
online_mode = m_demo ? "demo" : "offline";
|
online_mode = m_demo ? "demo" : "offline";
|
||||||
}
|
}
|
||||||
|
|
||||||
m_launcher->prependStep(
|
m_launcher->prependStep(makeShared<TextPrint>(m_launcher.get(), "Launched instance in " + online_mode + " mode\n", MessageLevel::Launcher));
|
||||||
makeShared<TextPrint>(m_launcher.get(), "Launched instance in " + online_mode + " mode\n", MessageLevel::Launcher));
|
|
||||||
|
|
||||||
// Prepend Version
|
// Prepend Version
|
||||||
{
|
{
|
||||||
auto versionString = QString("%1 version: %2 (%3)")
|
auto versionString = QString("%1 version: %2 (%3)").arg(BuildConfig.LAUNCHER_DISPLAYNAME, BuildConfig.printableVersionString(), BuildConfig.BUILD_PLATFORM);
|
||||||
.arg(BuildConfig.LAUNCHER_DISPLAYNAME, BuildConfig.printableVersionString(), BuildConfig.BUILD_PLATFORM);
|
|
||||||
m_launcher->prependStep(makeShared<TextPrint>(m_launcher.get(), versionString + "\n\n", MessageLevel::Launcher));
|
m_launcher->prependStep(makeShared<TextPrint>(m_launcher.get(), versionString + "\n\n", MessageLevel::Launcher));
|
||||||
}
|
}
|
||||||
m_launcher->start();
|
m_launcher->start();
|
||||||
@ -354,33 +399,37 @@ void LaunchController::launchInstance()
|
|||||||
|
|
||||||
void LaunchController::readyForLaunch()
|
void LaunchController::readyForLaunch()
|
||||||
{
|
{
|
||||||
if (!m_profiler) {
|
if (!m_profiler)
|
||||||
|
{
|
||||||
m_launcher->proceed();
|
m_launcher->proceed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
QString error;
|
QString error;
|
||||||
if (!m_profiler->check(&error)) {
|
if (!m_profiler->check(&error))
|
||||||
|
{
|
||||||
m_launcher->abort();
|
m_launcher->abort();
|
||||||
|
QMessageBox::critical(m_parentWidget, tr("Error!"), tr("Couldn't start profiler: %1").arg(error));
|
||||||
emitFailed("Profiler startup failed!");
|
emitFailed("Profiler startup failed!");
|
||||||
QMessageBox::critical(m_parentWidget, tr("Error!"), tr("Profiler check for %1 failed: %2").arg(m_profiler->name(), error));
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
BaseProfiler* profilerInstance = m_profiler->createProfiler(m_launcher->instance(), this);
|
BaseProfiler *profilerInstance = m_profiler->createProfiler(m_launcher->instance(), this);
|
||||||
|
|
||||||
connect(profilerInstance, &BaseProfiler::readyToLaunch, [this](const QString& message) {
|
connect(profilerInstance, &BaseProfiler::readyToLaunch, [this](const QString & message)
|
||||||
QMessageBox msg(m_parentWidget);
|
{
|
||||||
|
QMessageBox msg;
|
||||||
msg.setText(tr("The game launch is delayed until you press the "
|
msg.setText(tr("The game launch is delayed until you press the "
|
||||||
"button. This is the right time to setup the profiler, as the "
|
"button. This is the right time to setup the profiler, as the "
|
||||||
"profiler server is running now.\n\n%1")
|
"profiler server is running now.\n\n%1").arg(message));
|
||||||
.arg(message));
|
|
||||||
msg.setWindowTitle(tr("Waiting."));
|
msg.setWindowTitle(tr("Waiting."));
|
||||||
msg.setIcon(QMessageBox::Information);
|
msg.setIcon(QMessageBox::Information);
|
||||||
msg.addButton(tr("&Launch"), QMessageBox::AcceptRole);
|
msg.addButton(tr("Launch"), QMessageBox::AcceptRole);
|
||||||
|
msg.setModal(true);
|
||||||
msg.exec();
|
msg.exec();
|
||||||
m_launcher->proceed();
|
m_launcher->proceed();
|
||||||
});
|
});
|
||||||
connect(profilerInstance, &BaseProfiler::abortLaunch, [this](const QString& message) {
|
connect(profilerInstance, &BaseProfiler::abortLaunch, [this](const QString & message)
|
||||||
|
{
|
||||||
QMessageBox msg;
|
QMessageBox msg;
|
||||||
msg.setText(tr("Couldn't start the profiler: %1").arg(message));
|
msg.setText(tr("Couldn't start the profiler: %1").arg(message));
|
||||||
msg.setWindowTitle(tr("Error"));
|
msg.setWindowTitle(tr("Error"));
|
||||||
@ -401,7 +450,8 @@ void LaunchController::onSucceeded()
|
|||||||
|
|
||||||
void LaunchController::onFailed(QString reason)
|
void LaunchController::onFailed(QString reason)
|
||||||
{
|
{
|
||||||
if (m_instance->settings()->get("ShowConsoleOnError").toBool()) {
|
if(m_instance->settings()->get("ShowConsoleOnError").toBool())
|
||||||
|
{
|
||||||
APPLICATION->showInstanceWindow(m_instance, "console");
|
APPLICATION->showInstanceWindow(m_instance, "console");
|
||||||
}
|
}
|
||||||
emitFailed(reason);
|
emitFailed(reason);
|
||||||
@ -417,18 +467,21 @@ void LaunchController::onProgressRequested(Task* task)
|
|||||||
|
|
||||||
bool LaunchController::abort()
|
bool LaunchController::abort()
|
||||||
{
|
{
|
||||||
if (!m_launcher) {
|
if(!m_launcher)
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (!m_launcher->canAbort()) {
|
if(!m_launcher->canAbort())
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
auto response = CustomMessageBox::selectable(m_parentWidget, tr("Kill Minecraft?"),
|
auto response = CustomMessageBox::selectable(
|
||||||
tr("This can cause the instance to get corrupted and should only be used if Minecraft "
|
m_parentWidget, tr("Kill Minecraft?"),
|
||||||
"is frozen for some reason"),
|
tr("This can cause the instance to get corrupted and should only be used if Minecraft "
|
||||||
QMessageBox::Question, QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes)
|
"is frozen for some reason"),
|
||||||
->exec();
|
QMessageBox::Question, QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes)->exec();
|
||||||
if (response == QMessageBox::Yes) {
|
if (response == QMessageBox::Yes)
|
||||||
|
{
|
||||||
return m_launcher->abort();
|
return m_launcher->abort();
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
@ -34,61 +34,81 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
#include <QObject>
|
||||||
#include <BaseInstance.h>
|
#include <BaseInstance.h>
|
||||||
#include <tools/BaseProfiler.h>
|
#include <tools/BaseProfiler.h>
|
||||||
#include <QObject>
|
|
||||||
|
|
||||||
#include "minecraft/auth/MinecraftAccount.h"
|
|
||||||
#include "minecraft/launch/MinecraftServerTarget.h"
|
#include "minecraft/launch/MinecraftServerTarget.h"
|
||||||
|
#include "minecraft/auth/MinecraftAccount.h"
|
||||||
|
|
||||||
class InstanceWindow;
|
class InstanceWindow;
|
||||||
class LaunchController : public Task {
|
class LaunchController: public Task
|
||||||
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
void executeTask() override;
|
void executeTask() override;
|
||||||
|
|
||||||
LaunchController(QObject* parent = nullptr);
|
LaunchController(QObject * parent = nullptr);
|
||||||
virtual ~LaunchController(){};
|
virtual ~LaunchController(){};
|
||||||
|
|
||||||
void setInstance(InstancePtr instance) { m_instance = instance; }
|
void setInstance(InstancePtr instance) {
|
||||||
|
m_instance = instance;
|
||||||
|
}
|
||||||
|
|
||||||
InstancePtr instance() { return m_instance; }
|
InstancePtr instance() {
|
||||||
|
return m_instance;
|
||||||
|
}
|
||||||
|
|
||||||
void setOnline(bool online) { m_online = online; }
|
void setOnline(bool online) {
|
||||||
|
m_online = online;
|
||||||
|
}
|
||||||
|
|
||||||
void setDemo(bool demo) { m_demo = demo; }
|
void setDemo(bool demo) {
|
||||||
|
m_demo = demo;
|
||||||
|
}
|
||||||
|
|
||||||
void setProfiler(BaseProfilerFactory* profiler) { m_profiler = profiler; }
|
void setProfiler(BaseProfilerFactory *profiler) {
|
||||||
|
m_profiler = profiler;
|
||||||
|
}
|
||||||
|
|
||||||
void setParentWidget(QWidget* widget) { m_parentWidget = widget; }
|
void setParentWidget(QWidget * widget) {
|
||||||
|
m_parentWidget = widget;
|
||||||
|
}
|
||||||
|
|
||||||
void setServerToJoin(MinecraftServerTargetPtr serverToJoin) { m_serverToJoin = std::move(serverToJoin); }
|
void setServerToJoin(MinecraftServerTargetPtr serverToJoin) {
|
||||||
|
m_serverToJoin = std::move(serverToJoin);
|
||||||
|
}
|
||||||
|
|
||||||
void setAccountToUse(MinecraftAccountPtr accountToUse) { m_accountToUse = std::move(accountToUse); }
|
void setAccountToUse(MinecraftAccountPtr accountToUse) {
|
||||||
|
m_accountToUse = std::move(accountToUse);
|
||||||
|
}
|
||||||
|
|
||||||
QString id() { return m_instance->id(); }
|
QString id()
|
||||||
|
{
|
||||||
|
return m_instance->id();
|
||||||
|
}
|
||||||
|
|
||||||
bool abort() override;
|
bool abort() override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void login();
|
void login();
|
||||||
void launchInstance();
|
void launchInstance();
|
||||||
void decideAccount();
|
void decideAccount();
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void readyForLaunch();
|
void readyForLaunch();
|
||||||
|
|
||||||
void onSucceeded();
|
void onSucceeded();
|
||||||
void onFailed(QString reason);
|
void onFailed(QString reason);
|
||||||
void onProgressRequested(Task* task);
|
void onProgressRequested(Task *task);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
BaseProfilerFactory* m_profiler = nullptr;
|
BaseProfilerFactory *m_profiler = nullptr;
|
||||||
bool m_online = true;
|
bool m_online = true;
|
||||||
bool m_demo = false;
|
bool m_demo = false;
|
||||||
InstancePtr m_instance;
|
InstancePtr m_instance;
|
||||||
QWidget* m_parentWidget = nullptr;
|
QWidget * m_parentWidget = nullptr;
|
||||||
InstanceWindow* m_console = nullptr;
|
InstanceWindow *m_console = nullptr;
|
||||||
MinecraftAccountPtr m_accountToUse = nullptr;
|
MinecraftAccountPtr m_accountToUse = nullptr;
|
||||||
AuthSessionPtr m_session;
|
AuthSessionPtr m_session;
|
||||||
shared_qobject_ptr<LaunchTask> m_launcher;
|
shared_qobject_ptr<LaunchTask> m_launcher;
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* Prism Launcher - Minecraft Launcher
|
||||||
* Copyright (C) 2022,2023 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022,2023 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
* Copyright (c) 2023 flowln <flowlnlnln@gmail.com>
|
* Copyright (c) 2023 flowln <flowlnlnln@gmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -39,7 +39,7 @@
|
|||||||
#include <QTextDecoder>
|
#include <QTextDecoder>
|
||||||
#include "MessageLevel.h"
|
#include "MessageLevel.h"
|
||||||
|
|
||||||
LoggedProcess::LoggedProcess(QObject* parent) : QProcess(parent)
|
LoggedProcess::LoggedProcess(QObject *parent) : QProcess(parent)
|
||||||
{
|
{
|
||||||
// QProcess has a strange interface... let's map a lot of those into a few.
|
// QProcess has a strange interface... let's map a lot of those into a few.
|
||||||
connect(this, &QProcess::readyReadStandardOutput, this, &LoggedProcess::on_stdOut);
|
connect(this, &QProcess::readyReadStandardOutput, this, &LoggedProcess::on_stdOut);
|
||||||
@ -51,7 +51,8 @@ LoggedProcess::LoggedProcess(QObject* parent) : QProcess(parent)
|
|||||||
|
|
||||||
LoggedProcess::~LoggedProcess()
|
LoggedProcess::~LoggedProcess()
|
||||||
{
|
{
|
||||||
if (m_is_detachable) {
|
if(m_is_detachable)
|
||||||
|
{
|
||||||
setProcessState(QProcess::NotRunning);
|
setProcessState(QProcess::NotRunning);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -65,9 +66,14 @@ QStringList LoggedProcess::reprocess(const QByteArray& data, QTextDecoder& decod
|
|||||||
m_leftover_line = "";
|
m_leftover_line = "";
|
||||||
}
|
}
|
||||||
|
|
||||||
auto lines = str.remove(QChar::CarriageReturn).split(QChar::LineFeed);
|
#if QT_VERSION < QT_VERSION_CHECK(5, 14, 0)
|
||||||
|
auto lines = str.remove(QChar::CarriageReturn).split(QChar::LineFeed, QString::SkipEmptyParts);
|
||||||
|
#else
|
||||||
|
auto lines = str.remove(QChar::CarriageReturn).split(QChar::LineFeed, Qt::SkipEmptyParts);
|
||||||
|
#endif
|
||||||
|
|
||||||
m_leftover_line = lines.takeLast();
|
if (!str.endsWith(QChar::LineFeed))
|
||||||
|
m_leftover_line = lines.takeLast();
|
||||||
return lines;
|
return lines;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -89,31 +95,39 @@ void LoggedProcess::on_exit(int exit_code, QProcess::ExitStatus status)
|
|||||||
m_exit_code = exit_code;
|
m_exit_code = exit_code;
|
||||||
|
|
||||||
// based on state, send signals
|
// based on state, send signals
|
||||||
if (!m_is_aborting) {
|
if (!m_is_aborting)
|
||||||
if (status == QProcess::NormalExit) {
|
{
|
||||||
|
if (status == QProcess::NormalExit)
|
||||||
|
{
|
||||||
//: Message displayed on instance exit
|
//: Message displayed on instance exit
|
||||||
emit log({ tr("Process exited with code %1.").arg(exit_code) }, MessageLevel::Launcher);
|
emit log({tr("Process exited with code %1.").arg(exit_code)}, MessageLevel::Launcher);
|
||||||
changeState(LoggedProcess::Finished);
|
changeState(LoggedProcess::Finished);
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
//: Message displayed on instance crashed
|
//: Message displayed on instance crashed
|
||||||
if (exit_code == -1)
|
if(exit_code == -1)
|
||||||
emit log({ tr("Process crashed.") }, MessageLevel::Launcher);
|
emit log({tr("Process crashed.")}, MessageLevel::Launcher);
|
||||||
else
|
else
|
||||||
emit log({ tr("Process crashed with exitcode %1.").arg(exit_code) }, MessageLevel::Launcher);
|
emit log({tr("Process crashed with exitcode %1.").arg(exit_code)}, MessageLevel::Launcher);
|
||||||
changeState(LoggedProcess::Crashed);
|
changeState(LoggedProcess::Crashed);
|
||||||
}
|
}
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
//: Message displayed after the instance exits due to kill request
|
//: Message displayed after the instance exits due to kill request
|
||||||
emit log({ tr("Process was killed by user.") }, MessageLevel::Error);
|
emit log({tr("Process was killed by user.")}, MessageLevel::Error);
|
||||||
changeState(LoggedProcess::Aborted);
|
changeState(LoggedProcess::Aborted);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void LoggedProcess::on_error(QProcess::ProcessError error)
|
void LoggedProcess::on_error(QProcess::ProcessError error)
|
||||||
{
|
{
|
||||||
switch (error) {
|
switch(error)
|
||||||
case QProcess::FailedToStart: {
|
{
|
||||||
emit log({ tr("The process failed to start.") }, MessageLevel::Fatal);
|
case QProcess::FailedToStart:
|
||||||
|
{
|
||||||
|
emit log({tr("The process failed to start.")}, MessageLevel::Fatal);
|
||||||
changeState(LoggedProcess::FailedToStart);
|
changeState(LoggedProcess::FailedToStart);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -140,7 +154,7 @@ int LoggedProcess::exitCode() const
|
|||||||
|
|
||||||
void LoggedProcess::changeState(LoggedProcess::State state)
|
void LoggedProcess::changeState(LoggedProcess::State state)
|
||||||
{
|
{
|
||||||
if (state == m_state)
|
if(state == m_state)
|
||||||
return;
|
return;
|
||||||
m_state = state;
|
m_state = state;
|
||||||
emit stateChanged(m_state);
|
emit stateChanged(m_state);
|
||||||
@ -153,19 +167,24 @@ LoggedProcess::State LoggedProcess::state() const
|
|||||||
|
|
||||||
void LoggedProcess::on_stateChange(QProcess::ProcessState state)
|
void LoggedProcess::on_stateChange(QProcess::ProcessState state)
|
||||||
{
|
{
|
||||||
switch (state) {
|
switch(state)
|
||||||
|
{
|
||||||
case QProcess::NotRunning:
|
case QProcess::NotRunning:
|
||||||
break; // let's not - there are too many that handle this already.
|
break; // let's not - there are too many that handle this already.
|
||||||
case QProcess::Starting: {
|
case QProcess::Starting:
|
||||||
if (m_state != LoggedProcess::NotRunning) {
|
{
|
||||||
qWarning() << "Wrong state change for process from state" << m_state << "to" << (int)LoggedProcess::Starting;
|
if(m_state != LoggedProcess::NotRunning)
|
||||||
|
{
|
||||||
|
qWarning() << "Wrong state change for process from state" << m_state << "to" << (int) LoggedProcess::Starting;
|
||||||
}
|
}
|
||||||
changeState(LoggedProcess::Starting);
|
changeState(LoggedProcess::Starting);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
case QProcess::Running: {
|
case QProcess::Running:
|
||||||
if (m_state != LoggedProcess::Starting) {
|
{
|
||||||
qWarning() << "Wrong state change for process from state" << m_state << "to" << (int)LoggedProcess::Running;
|
if(m_state != LoggedProcess::Starting)
|
||||||
|
{
|
||||||
|
qWarning() << "Wrong state change for process from state" << m_state << "to" << (int) LoggedProcess::Running;
|
||||||
}
|
}
|
||||||
changeState(LoggedProcess::Running);
|
changeState(LoggedProcess::Running);
|
||||||
return;
|
return;
|
||||||
|
@ -43,12 +43,22 @@
|
|||||||
* This is a basic process.
|
* This is a basic process.
|
||||||
* It has line-based logging support and hides some of the nasty bits.
|
* It has line-based logging support and hides some of the nasty bits.
|
||||||
*/
|
*/
|
||||||
class LoggedProcess : public QProcess {
|
class LoggedProcess : public QProcess
|
||||||
Q_OBJECT
|
{
|
||||||
public:
|
Q_OBJECT
|
||||||
enum State { NotRunning, Starting, FailedToStart, Running, Finished, Crashed, Aborted };
|
public:
|
||||||
|
enum State
|
||||||
|
{
|
||||||
|
NotRunning,
|
||||||
|
Starting,
|
||||||
|
FailedToStart,
|
||||||
|
Running,
|
||||||
|
Finished,
|
||||||
|
Crashed,
|
||||||
|
Aborted
|
||||||
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit LoggedProcess(QObject* parent = 0);
|
explicit LoggedProcess(QObject* parent = 0);
|
||||||
virtual ~LoggedProcess();
|
virtual ~LoggedProcess();
|
||||||
|
|
||||||
@ -57,29 +67,30 @@ class LoggedProcess : public QProcess {
|
|||||||
|
|
||||||
void setDetachable(bool detachable);
|
void setDetachable(bool detachable);
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void log(QStringList lines, MessageLevel::Enum level);
|
void log(QStringList lines, MessageLevel::Enum level);
|
||||||
void stateChanged(LoggedProcess::State state);
|
void stateChanged(LoggedProcess::State state);
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
/**
|
/**
|
||||||
* @brief kill the process - equivalent to kill -9
|
* @brief kill the process - equivalent to kill -9
|
||||||
*/
|
*/
|
||||||
void kill();
|
void kill();
|
||||||
|
|
||||||
private slots:
|
|
||||||
|
private slots:
|
||||||
void on_stdErr();
|
void on_stdErr();
|
||||||
void on_stdOut();
|
void on_stdOut();
|
||||||
void on_exit(int exit_code, QProcess::ExitStatus status);
|
void on_exit(int exit_code, QProcess::ExitStatus status);
|
||||||
void on_error(QProcess::ProcessError error);
|
void on_error(QProcess::ProcessError error);
|
||||||
void on_stateChange(QProcess::ProcessState);
|
void on_stateChange(QProcess::ProcessState);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void changeState(LoggedProcess::State state);
|
void changeState(LoggedProcess::State state);
|
||||||
|
|
||||||
QStringList reprocess(const QByteArray& data, QTextDecoder& decoder);
|
QStringList reprocess(const QByteArray& data, QTextDecoder& decoder);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QTextDecoder m_err_decoder = QTextDecoder(QTextCodec::codecForLocale());
|
QTextDecoder m_err_decoder = QTextDecoder(QTextCodec::codecForLocale());
|
||||||
QTextDecoder m_out_decoder = QTextDecoder(QTextCodec::codecForLocale());
|
QTextDecoder m_out_decoder = QTextDecoder(QTextCodec::codecForLocale());
|
||||||
QString m_leftover_line;
|
QString m_leftover_line;
|
||||||
|
@ -16,31 +16,31 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <MMCTime.h>
|
#include <MMCTime.h>
|
||||||
#include <qobject.h>
|
|
||||||
|
|
||||||
#include <QDateTime>
|
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
|
#include <QDateTime>
|
||||||
#include <QTextStream>
|
#include <QTextStream>
|
||||||
|
|
||||||
QString Time::prettifyDuration(int64_t duration, bool noDays)
|
QString Time::prettifyDuration(int64_t duration) {
|
||||||
{
|
int seconds = (int) (duration % 60);
|
||||||
int seconds = (int)(duration % 60);
|
|
||||||
duration /= 60;
|
duration /= 60;
|
||||||
int minutes = (int)(duration % 60);
|
int minutes = (int) (duration % 60);
|
||||||
duration /= 60;
|
duration /= 60;
|
||||||
int hours = (int)(noDays ? duration : (duration % 24));
|
int hours = (int) (duration % 24);
|
||||||
int days = (int)(noDays ? 0 : (duration / 24));
|
int days = (int) (duration / 24);
|
||||||
if ((hours == 0) && (days == 0)) {
|
if((hours == 0)&&(days == 0))
|
||||||
|
{
|
||||||
return QObject::tr("%1min %2s").arg(minutes).arg(seconds);
|
return QObject::tr("%1min %2s").arg(minutes).arg(seconds);
|
||||||
}
|
}
|
||||||
if (days == 0) {
|
if (days == 0)
|
||||||
|
{
|
||||||
return QObject::tr("%1h %2min").arg(hours).arg(minutes);
|
return QObject::tr("%1h %2min").arg(hours).arg(minutes);
|
||||||
}
|
}
|
||||||
return QObject::tr("%1d %2h %3min").arg(days).arg(hours).arg(minutes);
|
return QObject::tr("%1d %2h %3min").arg(days).arg(hours).arg(minutes);
|
||||||
}
|
}
|
||||||
|
|
||||||
QString Time::humanReadableDuration(double duration, int precision)
|
QString Time::humanReadableDuration(double duration, int precision) {
|
||||||
{
|
|
||||||
using days = std::chrono::duration<int, std::ratio<86400>>;
|
using days = std::chrono::duration<int, std::ratio<86400>>;
|
||||||
|
|
||||||
QString outStr;
|
QString outStr;
|
||||||
@ -48,10 +48,10 @@ QString Time::humanReadableDuration(double duration, int precision)
|
|||||||
|
|
||||||
bool neg = false;
|
bool neg = false;
|
||||||
if (duration < 0) {
|
if (duration < 0) {
|
||||||
neg = true; // flag
|
neg = true; // flag
|
||||||
duration *= -1; // invert
|
duration *= -1; // invert
|
||||||
}
|
}
|
||||||
|
|
||||||
auto std_duration = std::chrono::duration<double>(duration);
|
auto std_duration = std::chrono::duration<double>(duration);
|
||||||
auto d = std::chrono::duration_cast<days>(std_duration);
|
auto d = std::chrono::duration_cast<days>(std_duration);
|
||||||
std_duration -= d;
|
std_duration -= d;
|
||||||
@ -78,22 +78,22 @@ QString Time::humanReadableDuration(double duration, int precision)
|
|||||||
if (hc) {
|
if (hc) {
|
||||||
if (dc)
|
if (dc)
|
||||||
os << " ";
|
os << " ";
|
||||||
os << qSetFieldWidth(2) << hc << QObject::tr("h"); // hours
|
os << qSetFieldWidth(2) << hc << QObject::tr("h"); // hours
|
||||||
}
|
}
|
||||||
if (mc) {
|
if (mc) {
|
||||||
if (dc || hc)
|
if (dc || hc)
|
||||||
os << " ";
|
os << " ";
|
||||||
os << qSetFieldWidth(2) << mc << QObject::tr("m"); // minutes
|
os << qSetFieldWidth(2) << mc << QObject::tr("m"); // minutes
|
||||||
}
|
}
|
||||||
if (dc || hc || mc || sc) {
|
if (dc || hc || mc || sc) {
|
||||||
if (dc || hc || mc)
|
if (dc || hc || mc)
|
||||||
os << " ";
|
os << " ";
|
||||||
os << qSetFieldWidth(2) << sc << QObject::tr("s"); // seconds
|
os << qSetFieldWidth(2) << sc << QObject::tr("s"); // seconds
|
||||||
}
|
}
|
||||||
if ((msc && (precision > 0)) || !(dc || hc || mc || sc)) {
|
if ((msc && (precision > 0)) || !(dc || hc || mc || sc)) {
|
||||||
if (dc || hc || mc || sc)
|
if (dc || hc || mc || sc)
|
||||||
os << " ";
|
os << " ";
|
||||||
os << qSetFieldWidth(0) << qSetRealNumberPrecision(precision) << msc << QObject::tr("ms"); // miliseconds
|
os << qSetFieldWidth(0) << qSetRealNumberPrecision(precision) << msc << QObject::tr("ms"); // miliseconds
|
||||||
}
|
}
|
||||||
|
|
||||||
os.flush();
|
os.flush();
|
||||||
|
@ -20,15 +20,15 @@
|
|||||||
|
|
||||||
namespace Time {
|
namespace Time {
|
||||||
|
|
||||||
QString prettifyDuration(int64_t duration, bool noDays = false);
|
QString prettifyDuration(int64_t duration);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Returns a string with short form time duration ie. `2days 1h3m4s56.0ms`.
|
* @brief Returns a string with short form time duration ie. `2days 1h3m4s56.0ms`.
|
||||||
* miliseconds are only included if `precision` is greater than 0.
|
* miliseconds are only included if `precision` is greater than 0.
|
||||||
*
|
*
|
||||||
* @param duration a number of seconds as floating point
|
* @param duration a number of seconds as floating point
|
||||||
* @param precision number of decmial points to display on fractons of a second, defualts to 0.
|
* @param precision number of decmial points to display on fractons of a second, defualts to 0.
|
||||||
* @return QString
|
* @return QString
|
||||||
*/
|
*/
|
||||||
QString humanReadableDuration(double duration, int precision = 0);
|
QString humanReadableDuration(double duration, int precision = 0);
|
||||||
} // namespace Time
|
}
|
||||||
|
@ -1,8 +1,7 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
* Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com>
|
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -34,54 +33,56 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "MMCZip.h"
|
|
||||||
#include <quazip/quazip.h>
|
#include <quazip/quazip.h>
|
||||||
#include <quazip/quazipdir.h>
|
#include <quazip/quazipdir.h>
|
||||||
#include <quazip/quazipfile.h>
|
#include <quazip/quazipfile.h>
|
||||||
|
#include "MMCZip.h"
|
||||||
#include "FileSystem.h"
|
#include "FileSystem.h"
|
||||||
|
|
||||||
#include <QCoreApplication>
|
#include <QCoreApplication>
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
#include <QUrl>
|
|
||||||
|
|
||||||
#if defined(LAUNCHER_APPLICATION)
|
|
||||||
#include <QtConcurrentRun>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace MMCZip {
|
|
||||||
// ours
|
// ours
|
||||||
bool mergeZipFiles(QuaZip* into, QFileInfo from, QSet<QString>& contained, const FilterFunction filter)
|
bool MMCZip::mergeZipFiles(QuaZip *into, QFileInfo from, QSet<QString> &contained, const FilterFunction filter)
|
||||||
{
|
{
|
||||||
QuaZip modZip(from.filePath());
|
QuaZip modZip(from.filePath());
|
||||||
modZip.open(QuaZip::mdUnzip);
|
modZip.open(QuaZip::mdUnzip);
|
||||||
|
|
||||||
QuaZipFile fileInsideMod(&modZip);
|
QuaZipFile fileInsideMod(&modZip);
|
||||||
QuaZipFile zipOutFile(into);
|
QuaZipFile zipOutFile(into);
|
||||||
for (bool more = modZip.goToFirstFile(); more; more = modZip.goToNextFile()) {
|
for (bool more = modZip.goToFirstFile(); more; more = modZip.goToNextFile())
|
||||||
|
{
|
||||||
QString filename = modZip.getCurrentFileName();
|
QString filename = modZip.getCurrentFileName();
|
||||||
if (filter && !filter(filename)) {
|
if (filter && !filter(filename))
|
||||||
qDebug() << "Skipping file " << filename << " from " << from.fileName() << " - filtered";
|
{
|
||||||
|
qDebug() << "Skipping file " << filename << " from "
|
||||||
|
<< from.fileName() << " - filtered";
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if (contained.contains(filename)) {
|
if (contained.contains(filename))
|
||||||
qDebug() << "Skipping already contained file " << filename << " from " << from.fileName();
|
{
|
||||||
|
qDebug() << "Skipping already contained file " << filename << " from "
|
||||||
|
<< from.fileName();
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
contained.insert(filename);
|
contained.insert(filename);
|
||||||
|
|
||||||
if (!fileInsideMod.open(QIODevice::ReadOnly)) {
|
if (!fileInsideMod.open(QIODevice::ReadOnly))
|
||||||
|
{
|
||||||
qCritical() << "Failed to open " << filename << " from " << from.fileName();
|
qCritical() << "Failed to open " << filename << " from " << from.fileName();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
QuaZipNewInfo info_out(fileInsideMod.getActualFileName());
|
QuaZipNewInfo info_out(fileInsideMod.getActualFileName());
|
||||||
|
|
||||||
if (!zipOutFile.open(QIODevice::WriteOnly, info_out)) {
|
if (!zipOutFile.open(QIODevice::WriteOnly, info_out))
|
||||||
|
{
|
||||||
qCritical() << "Failed to open " << filename << " in the jar";
|
qCritical() << "Failed to open " << filename << " in the jar";
|
||||||
fileInsideMod.close();
|
fileInsideMod.close();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (!JlCompress::copyData(fileInsideMod, zipOutFile)) {
|
if (!JlCompress::copyData(fileInsideMod, zipOutFile))
|
||||||
|
{
|
||||||
zipOutFile.close();
|
zipOutFile.close();
|
||||||
fileInsideMod.close();
|
fileInsideMod.close();
|
||||||
qCritical() << "Failed to copy data of " << filename << " into the jar";
|
qCritical() << "Failed to copy data of " << filename << " into the jar";
|
||||||
@ -93,11 +94,10 @@ bool mergeZipFiles(QuaZip* into, QFileInfo from, QSet<QString>& contained, const
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool compressDirFiles(QuaZip* zip, QString dir, QFileInfoList files, bool followSymlinks)
|
bool MMCZip::compressDirFiles(QuaZip *zip, QString dir, QFileInfoList files, bool followSymlinks)
|
||||||
{
|
{
|
||||||
QDir directory(dir);
|
QDir directory(dir);
|
||||||
if (!directory.exists())
|
if (!directory.exists()) return false;
|
||||||
return false;
|
|
||||||
|
|
||||||
for (auto e : files) {
|
for (auto e : files) {
|
||||||
auto filePath = directory.relativeFilePath(e.absoluteFilePath());
|
auto filePath = directory.relativeFilePath(e.absoluteFilePath());
|
||||||
@ -109,18 +109,17 @@ bool compressDirFiles(QuaZip* zip, QString dir, QFileInfoList files, bool follow
|
|||||||
srcPath = e.canonicalFilePath();
|
srcPath = e.canonicalFilePath();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!JlCompress::compressFile(zip, srcPath, filePath))
|
if( !JlCompress::compressFile(zip, srcPath, filePath)) return false;
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool compressDirFiles(QString fileCompressed, QString dir, QFileInfoList files, bool followSymlinks)
|
bool MMCZip::compressDirFiles(QString fileCompressed, QString dir, QFileInfoList files, bool followSymlinks)
|
||||||
{
|
{
|
||||||
QuaZip zip(fileCompressed);
|
QuaZip zip(fileCompressed);
|
||||||
QDir().mkpath(QFileInfo(fileCompressed).absolutePath());
|
QDir().mkpath(QFileInfo(fileCompressed).absolutePath());
|
||||||
if (!zip.open(QuaZip::mdCreate)) {
|
if(!zip.open(QuaZip::mdCreate)) {
|
||||||
QFile::remove(fileCompressed);
|
QFile::remove(fileCompressed);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -128,7 +127,7 @@ bool compressDirFiles(QString fileCompressed, QString dir, QFileInfoList files,
|
|||||||
auto result = compressDirFiles(&zip, dir, files, followSymlinks);
|
auto result = compressDirFiles(&zip, dir, files, followSymlinks);
|
||||||
|
|
||||||
zip.close();
|
zip.close();
|
||||||
if (zip.getZipError() != 0) {
|
if(zip.getZipError()!=0) {
|
||||||
QFile::remove(fileCompressed);
|
QFile::remove(fileCompressed);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -136,12 +135,12 @@ bool compressDirFiles(QString fileCompressed, QString dir, QFileInfoList files,
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(LAUNCHER_APPLICATION)
|
|
||||||
// ours
|
// ours
|
||||||
bool createModdedJar(QString sourceJarPath, QString targetJarPath, const QList<Mod*>& mods)
|
bool MMCZip::createModdedJar(QString sourceJarPath, QString targetJarPath, const QList<Mod*>& mods)
|
||||||
{
|
{
|
||||||
QuaZip zipOut(targetJarPath);
|
QuaZip zipOut(targetJarPath);
|
||||||
if (!zipOut.open(QuaZip::mdCreate)) {
|
if (!zipOut.open(QuaZip::mdCreate))
|
||||||
|
{
|
||||||
QFile::remove(targetJarPath);
|
QFile::remove(targetJarPath);
|
||||||
qCritical() << "Failed to open the minecraft.jar for modding";
|
qCritical() << "Failed to open the minecraft.jar for modding";
|
||||||
return false;
|
return false;
|
||||||
@ -152,29 +151,37 @@ bool createModdedJar(QString sourceJarPath, QString targetJarPath, const QList<M
|
|||||||
|
|
||||||
// Modify the jar
|
// Modify the jar
|
||||||
// This needs to be done in reverse-order to ensure we respect the loading order of components
|
// This needs to be done in reverse-order to ensure we respect the loading order of components
|
||||||
for (auto i = mods.crbegin(); i != mods.crend(); i++) {
|
for (auto i = mods.crbegin(); i != mods.crend(); i++)
|
||||||
|
{
|
||||||
const auto* mod = *i;
|
const auto* mod = *i;
|
||||||
// do not merge disabled mods.
|
// do not merge disabled mods.
|
||||||
if (!mod->enabled())
|
if (!mod->enabled())
|
||||||
continue;
|
continue;
|
||||||
if (mod->type() == ResourceType::ZIPFILE) {
|
if (mod->type() == ResourceType::ZIPFILE)
|
||||||
if (!mergeZipFiles(&zipOut, mod->fileinfo(), addedFiles)) {
|
{
|
||||||
|
if (!mergeZipFiles(&zipOut, mod->fileinfo(), addedFiles))
|
||||||
|
{
|
||||||
zipOut.close();
|
zipOut.close();
|
||||||
QFile::remove(targetJarPath);
|
QFile::remove(targetJarPath);
|
||||||
qCritical() << "Failed to add" << mod->fileinfo().fileName() << "to the jar.";
|
qCritical() << "Failed to add" << mod->fileinfo().fileName() << "to the jar.";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
} else if (mod->type() == ResourceType::SINGLEFILE) {
|
}
|
||||||
|
else if (mod->type() == ResourceType::SINGLEFILE)
|
||||||
|
{
|
||||||
// FIXME: buggy - does not work with addedFiles
|
// FIXME: buggy - does not work with addedFiles
|
||||||
auto filename = mod->fileinfo();
|
auto filename = mod->fileinfo();
|
||||||
if (!JlCompress::compressFile(&zipOut, filename.absoluteFilePath(), filename.fileName())) {
|
if (!JlCompress::compressFile(&zipOut, filename.absoluteFilePath(), filename.fileName()))
|
||||||
|
{
|
||||||
zipOut.close();
|
zipOut.close();
|
||||||
QFile::remove(targetJarPath);
|
QFile::remove(targetJarPath);
|
||||||
qCritical() << "Failed to add" << mod->fileinfo().fileName() << "to the jar.";
|
qCritical() << "Failed to add" << mod->fileinfo().fileName() << "to the jar.";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
addedFiles.insert(filename.fileName());
|
addedFiles.insert(filename.fileName());
|
||||||
} else if (mod->type() == ResourceType::FOLDER) {
|
}
|
||||||
|
else if (mod->type() == ResourceType::FOLDER)
|
||||||
|
{
|
||||||
// untested, but seems to be unused / not possible to reach
|
// untested, but seems to be unused / not possible to reach
|
||||||
// FIXME: buggy - does not work with addedFiles
|
// FIXME: buggy - does not work with addedFiles
|
||||||
auto filename = mod->fileinfo();
|
auto filename = mod->fileinfo();
|
||||||
@ -183,21 +190,25 @@ bool createModdedJar(QString sourceJarPath, QString targetJarPath, const QList<M
|
|||||||
dir.cdUp();
|
dir.cdUp();
|
||||||
QString parent_dir = dir.absolutePath();
|
QString parent_dir = dir.absolutePath();
|
||||||
auto files = QFileInfoList();
|
auto files = QFileInfoList();
|
||||||
collectFileListRecursively(what_to_zip, nullptr, &files, nullptr);
|
MMCZip::collectFileListRecursively(what_to_zip, nullptr, &files, nullptr);
|
||||||
|
|
||||||
for (auto e : files) {
|
for (auto e : files) {
|
||||||
if (addedFiles.contains(e.filePath()))
|
if (addedFiles.contains(e.filePath()))
|
||||||
files.removeAll(e);
|
files.removeAll(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!compressDirFiles(&zipOut, parent_dir, files)) {
|
if (!MMCZip::compressDirFiles(&zipOut, parent_dir, files))
|
||||||
|
{
|
||||||
zipOut.close();
|
zipOut.close();
|
||||||
QFile::remove(targetJarPath);
|
QFile::remove(targetJarPath);
|
||||||
qCritical() << "Failed to add" << mod->fileinfo().fileName() << "to the jar.";
|
qCritical() << "Failed to add" << mod->fileinfo().fileName() << "to the jar.";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
qDebug() << "Adding folder " << filename.fileName() << " from " << filename.absoluteFilePath();
|
qDebug() << "Adding folder " << filename.fileName() << " from "
|
||||||
} else {
|
<< filename.absoluteFilePath();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
// Make sure we do not continue launching when something is missing or undefined...
|
// Make sure we do not continue launching when something is missing or undefined...
|
||||||
zipOut.close();
|
zipOut.close();
|
||||||
QFile::remove(targetJarPath);
|
QFile::remove(targetJarPath);
|
||||||
@ -206,7 +217,8 @@ bool createModdedJar(QString sourceJarPath, QString targetJarPath, const QList<M
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!mergeZipFiles(&zipOut, QFileInfo(sourceJarPath), addedFiles, [](const QString key) { return !key.contains("META-INF"); })) {
|
if (!mergeZipFiles(&zipOut, QFileInfo(sourceJarPath), addedFiles, [](const QString key){return !key.contains("META-INF");}))
|
||||||
|
{
|
||||||
zipOut.close();
|
zipOut.close();
|
||||||
QFile::remove(targetJarPath);
|
QFile::remove(targetJarPath);
|
||||||
qCritical() << "Failed to insert minecraft.jar contents.";
|
qCritical() << "Failed to insert minecraft.jar contents.";
|
||||||
@ -215,17 +227,17 @@ bool createModdedJar(QString sourceJarPath, QString targetJarPath, const QList<M
|
|||||||
|
|
||||||
// Recompress the jar
|
// Recompress the jar
|
||||||
zipOut.close();
|
zipOut.close();
|
||||||
if (zipOut.getZipError() != 0) {
|
if (zipOut.getZipError() != 0)
|
||||||
|
{
|
||||||
QFile::remove(targetJarPath);
|
QFile::remove(targetJarPath);
|
||||||
qCritical() << "Failed to finalize minecraft.jar!";
|
qCritical() << "Failed to finalize minecraft.jar!";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
// ours
|
// ours
|
||||||
QString findFolderOfFileInZip(QuaZip* zip, const QString& what, const QStringList& ignore_paths, const QString& root)
|
QString MMCZip::findFolderOfFileInZip(QuaZip* zip, const QString& what, const QStringList& ignore_paths, const QString& root)
|
||||||
{
|
{
|
||||||
QuaZipDir rootDir(zip, root);
|
QuaZipDir rootDir(zip, root);
|
||||||
for (auto&& fileName : rootDir.entryList(QDir::Files)) {
|
for (auto&& fileName : rootDir.entryList(QDir::Files)) {
|
||||||
@ -249,23 +261,27 @@ QString findFolderOfFileInZip(QuaZip* zip, const QString& what, const QStringLis
|
|||||||
}
|
}
|
||||||
|
|
||||||
// ours
|
// ours
|
||||||
bool findFilesInZip(QuaZip* zip, const QString& what, QStringList& result, const QString& root)
|
bool MMCZip::findFilesInZip(QuaZip * zip, const QString & what, QStringList & result, const QString &root)
|
||||||
{
|
{
|
||||||
QuaZipDir rootDir(zip, root);
|
QuaZipDir rootDir(zip, root);
|
||||||
for (auto fileName : rootDir.entryList(QDir::Files)) {
|
for(auto fileName: rootDir.entryList(QDir::Files))
|
||||||
if (fileName == what) {
|
{
|
||||||
|
if(fileName == what)
|
||||||
|
{
|
||||||
result.append(root);
|
result.append(root);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (auto fileName : rootDir.entryList(QDir::Dirs)) {
|
for(auto fileName: rootDir.entryList(QDir::Dirs))
|
||||||
|
{
|
||||||
findFilesInZip(zip, what, result, root + fileName);
|
findFilesInZip(zip, what, result, root + fileName);
|
||||||
}
|
}
|
||||||
return !result.isEmpty();
|
return !result.isEmpty();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// ours
|
// ours
|
||||||
std::optional<QStringList> extractSubDir(QuaZip* zip, const QString& subdir, const QString& target)
|
std::optional<QStringList> MMCZip::extractSubDir(QuaZip *zip, const QString & subdir, const QString &target)
|
||||||
{
|
{
|
||||||
auto target_top_dir = QUrl::fromLocalFile(target);
|
auto target_top_dir = QUrl::fromLocalFile(target);
|
||||||
|
|
||||||
@ -273,13 +289,16 @@ std::optional<QStringList> extractSubDir(QuaZip* zip, const QString& subdir, con
|
|||||||
|
|
||||||
qDebug() << "Extracting subdir" << subdir << "from" << zip->getZipName() << "to" << target;
|
qDebug() << "Extracting subdir" << subdir << "from" << zip->getZipName() << "to" << target;
|
||||||
auto numEntries = zip->getEntriesCount();
|
auto numEntries = zip->getEntriesCount();
|
||||||
if (numEntries < 0) {
|
if(numEntries < 0) {
|
||||||
qWarning() << "Failed to enumerate files in archive";
|
qWarning() << "Failed to enumerate files in archive";
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
} else if (numEntries == 0) {
|
}
|
||||||
|
else if(numEntries == 0) {
|
||||||
qDebug() << "Extracting empty archives seems odd...";
|
qDebug() << "Extracting empty archives seems odd...";
|
||||||
return extracted;
|
return extracted;
|
||||||
} else if (!zip->goToFirstFile()) {
|
}
|
||||||
|
else if (!zip->goToFirstFile())
|
||||||
|
{
|
||||||
qWarning() << "Failed to seek to first file in zip";
|
qWarning() << "Failed to seek to first file in zip";
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
@ -315,8 +334,7 @@ std::optional<QStringList> extractSubDir(QuaZip* zip, const QString& subdir, con
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!target_top_dir.isParentOf(QUrl::fromLocalFile(target_file_path))) {
|
if (!target_top_dir.isParentOf(QUrl::fromLocalFile(target_file_path))) {
|
||||||
qWarning() << "Extracting" << relative_file_name << "was cancelled, because it was effectively outside of the target path"
|
qWarning() << "Extracting" << relative_file_name << "was cancelled, because it was effectively outside of the target path" << target;
|
||||||
<< target;
|
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -327,8 +345,7 @@ std::optional<QStringList> extractSubDir(QuaZip* zip, const QString& subdir, con
|
|||||||
}
|
}
|
||||||
|
|
||||||
extracted.append(target_file_path);
|
extracted.append(target_file_path);
|
||||||
QFile::setPermissions(target_file_path,
|
QFile::setPermissions(target_file_path, QFileDevice::Permission::ReadUser | QFileDevice::Permission::WriteUser | QFileDevice::Permission::ExeUser);
|
||||||
QFileDevice::Permission::ReadUser | QFileDevice::Permission::WriteUser | QFileDevice::Permission::ExeUser);
|
|
||||||
|
|
||||||
qDebug() << "Extracted file" << relative_file_name << "to" << target_file_path;
|
qDebug() << "Extracted file" << relative_file_name << "to" << target_file_path;
|
||||||
} while (zip->goToNextFile());
|
} while (zip->goToNextFile());
|
||||||
@ -337,66 +354,66 @@ std::optional<QStringList> extractSubDir(QuaZip* zip, const QString& subdir, con
|
|||||||
}
|
}
|
||||||
|
|
||||||
// ours
|
// ours
|
||||||
bool extractRelFile(QuaZip* zip, const QString& file, const QString& target)
|
bool MMCZip::extractRelFile(QuaZip *zip, const QString &file, const QString &target)
|
||||||
{
|
{
|
||||||
return JlCompress::extractFile(zip, file, target);
|
return JlCompress::extractFile(zip, file, target);
|
||||||
}
|
}
|
||||||
|
|
||||||
// ours
|
// ours
|
||||||
std::optional<QStringList> extractDir(QString fileCompressed, QString dir)
|
std::optional<QStringList> MMCZip::extractDir(QString fileCompressed, QString dir)
|
||||||
{
|
{
|
||||||
QuaZip zip(fileCompressed);
|
QuaZip zip(fileCompressed);
|
||||||
if (!zip.open(QuaZip::mdUnzip)) {
|
if (!zip.open(QuaZip::mdUnzip))
|
||||||
|
{
|
||||||
// check if this is a minimum size empty zip file...
|
// check if this is a minimum size empty zip file...
|
||||||
QFileInfo fileInfo(fileCompressed);
|
QFileInfo fileInfo(fileCompressed);
|
||||||
if (fileInfo.size() == 22) {
|
if(fileInfo.size() == 22) {
|
||||||
return QStringList();
|
return QStringList();
|
||||||
}
|
}
|
||||||
qWarning() << "Could not open archive for unzipping:" << fileCompressed << "Error:" << zip.getZipError();
|
qWarning() << "Could not open archive for unzipping:" << fileCompressed << "Error:" << zip.getZipError();;
|
||||||
;
|
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
return extractSubDir(&zip, "", dir);
|
return MMCZip::extractSubDir(&zip, "", dir);
|
||||||
}
|
}
|
||||||
|
|
||||||
// ours
|
// ours
|
||||||
std::optional<QStringList> extractDir(QString fileCompressed, QString subdir, QString dir)
|
std::optional<QStringList> MMCZip::extractDir(QString fileCompressed, QString subdir, QString dir)
|
||||||
{
|
{
|
||||||
QuaZip zip(fileCompressed);
|
QuaZip zip(fileCompressed);
|
||||||
if (!zip.open(QuaZip::mdUnzip)) {
|
if (!zip.open(QuaZip::mdUnzip))
|
||||||
|
{
|
||||||
// check if this is a minimum size empty zip file...
|
// check if this is a minimum size empty zip file...
|
||||||
QFileInfo fileInfo(fileCompressed);
|
QFileInfo fileInfo(fileCompressed);
|
||||||
if (fileInfo.size() == 22) {
|
if(fileInfo.size() == 22) {
|
||||||
return QStringList();
|
return QStringList();
|
||||||
}
|
}
|
||||||
qWarning() << "Could not open archive for unzipping:" << fileCompressed << "Error:" << zip.getZipError();
|
qWarning() << "Could not open archive for unzipping:" << fileCompressed << "Error:" << zip.getZipError();;
|
||||||
;
|
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
return extractSubDir(&zip, subdir, dir);
|
return MMCZip::extractSubDir(&zip, subdir, dir);
|
||||||
}
|
}
|
||||||
|
|
||||||
// ours
|
// ours
|
||||||
bool extractFile(QString fileCompressed, QString file, QString target)
|
bool MMCZip::extractFile(QString fileCompressed, QString file, QString target)
|
||||||
{
|
{
|
||||||
QuaZip zip(fileCompressed);
|
QuaZip zip(fileCompressed);
|
||||||
if (!zip.open(QuaZip::mdUnzip)) {
|
if (!zip.open(QuaZip::mdUnzip))
|
||||||
|
{
|
||||||
// check if this is a minimum size empty zip file...
|
// check if this is a minimum size empty zip file...
|
||||||
QFileInfo fileInfo(fileCompressed);
|
QFileInfo fileInfo(fileCompressed);
|
||||||
if (fileInfo.size() == 22) {
|
if(fileInfo.size() == 22) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
qWarning() << "Could not open archive for unzipping:" << fileCompressed << "Error:" << zip.getZipError();
|
qWarning() << "Could not open archive for unzipping:" << fileCompressed << "Error:" << zip.getZipError();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return extractRelFile(&zip, file, target);
|
return MMCZip::extractRelFile(&zip, file, target);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool collectFileListRecursively(const QString& rootDir, const QString& subDir, QFileInfoList* files, FilterFunction excludeFilter)
|
bool MMCZip::collectFileListRecursively(const QString& rootDir, const QString& subDir, QFileInfoList *files,
|
||||||
{
|
MMCZip::FilterFunction excludeFilter) {
|
||||||
QDir rootDirectory(rootDir);
|
QDir rootDirectory(rootDir);
|
||||||
if (!rootDirectory.exists())
|
if (!rootDirectory.exists()) return false;
|
||||||
return false;
|
|
||||||
|
|
||||||
QDir directory;
|
QDir directory;
|
||||||
if (subDir == nullptr)
|
if (subDir == nullptr)
|
||||||
@ -404,109 +421,25 @@ bool collectFileListRecursively(const QString& rootDir, const QString& subDir, Q
|
|||||||
else
|
else
|
||||||
directory = QDir(subDir);
|
directory = QDir(subDir);
|
||||||
|
|
||||||
if (!directory.exists())
|
if (!directory.exists()) return false; // shouldn't ever happen
|
||||||
return false; // shouldn't ever happen
|
|
||||||
|
|
||||||
// recurse directories
|
// recurse directories
|
||||||
QFileInfoList entries = directory.entryInfoList(QDir::AllDirs | QDir::NoDotAndDotDot | QDir::Hidden);
|
QFileInfoList entries = directory.entryInfoList(QDir::AllDirs | QDir::NoDotAndDotDot | QDir::Hidden);
|
||||||
for (const auto& e : entries) {
|
for (const auto& e: entries) {
|
||||||
if (!collectFileListRecursively(rootDir, e.filePath(), files, excludeFilter))
|
if (!collectFileListRecursively(rootDir, e.filePath(), files, excludeFilter))
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// collect files
|
// collect files
|
||||||
entries = directory.entryInfoList(QDir::Files);
|
entries = directory.entryInfoList(QDir::Files);
|
||||||
for (const auto& e : entries) {
|
for (const auto& e: entries) {
|
||||||
QString relativeFilePath = rootDirectory.relativeFilePath(e.absoluteFilePath());
|
QString relativeFilePath = rootDirectory.relativeFilePath(e.absoluteFilePath());
|
||||||
if (excludeFilter && excludeFilter(relativeFilePath)) {
|
if (excludeFilter && excludeFilter(relativeFilePath)) {
|
||||||
qDebug() << "Skipping file " << relativeFilePath;
|
qDebug() << "Skipping file " << relativeFilePath;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
files->append(e); // we want the original paths for compressDirFiles
|
files->append(e); // we want the original paths for MMCZip::compressDirFiles
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(LAUNCHER_APPLICATION)
|
|
||||||
void ExportToZipTask::executeTask()
|
|
||||||
{
|
|
||||||
setStatus("Adding files...");
|
|
||||||
setProgress(0, m_files.length());
|
|
||||||
m_build_zip_future = QtConcurrent::run(QThreadPool::globalInstance(), [this]() { return exportZip(); });
|
|
||||||
connect(&m_build_zip_watcher, &QFutureWatcher<ZipResult>::finished, this, &ExportToZipTask::finish);
|
|
||||||
m_build_zip_watcher.setFuture(m_build_zip_future);
|
|
||||||
}
|
|
||||||
|
|
||||||
auto ExportToZipTask::exportZip() -> ZipResult
|
|
||||||
{
|
|
||||||
if (!m_dir.exists()) {
|
|
||||||
return ZipResult(tr("Folder doesn't exist"));
|
|
||||||
}
|
|
||||||
if (!m_output.isOpen() && !m_output.open(QuaZip::mdCreate)) {
|
|
||||||
return ZipResult(tr("Could not create file"));
|
|
||||||
}
|
|
||||||
|
|
||||||
for (auto fileName : m_extra_files.keys()) {
|
|
||||||
if (m_build_zip_future.isCanceled())
|
|
||||||
return ZipResult();
|
|
||||||
QuaZipFile indexFile(&m_output);
|
|
||||||
if (!indexFile.open(QIODevice::WriteOnly, QuaZipNewInfo(fileName))) {
|
|
||||||
return ZipResult(tr("Could not create:") + fileName);
|
|
||||||
}
|
|
||||||
indexFile.write(m_extra_files[fileName]);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const QFileInfo& file : m_files) {
|
|
||||||
if (m_build_zip_future.isCanceled())
|
|
||||||
return ZipResult();
|
|
||||||
|
|
||||||
auto absolute = file.absoluteFilePath();
|
|
||||||
auto relative = m_dir.relativeFilePath(absolute);
|
|
||||||
setStatus("Compresing: " + relative);
|
|
||||||
setProgress(m_progress + 1, m_progressTotal);
|
|
||||||
if (m_follow_symlinks) {
|
|
||||||
if (file.isSymLink())
|
|
||||||
absolute = file.symLinkTarget();
|
|
||||||
else
|
|
||||||
absolute = file.canonicalFilePath();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!m_exclude_files.contains(relative) && !JlCompress::compressFile(&m_output, absolute, m_destination_prefix + relative)) {
|
|
||||||
return ZipResult(tr("Could not read and compress %1").arg(relative));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
m_output.close();
|
|
||||||
if (m_output.getZipError() != 0) {
|
|
||||||
return ZipResult(tr("A zip error occurred"));
|
|
||||||
}
|
|
||||||
return ZipResult();
|
|
||||||
}
|
|
||||||
|
|
||||||
void ExportToZipTask::finish()
|
|
||||||
{
|
|
||||||
if (m_build_zip_future.isCanceled()) {
|
|
||||||
QFile::remove(m_output_path);
|
|
||||||
emitAborted();
|
|
||||||
} else if (auto result = m_build_zip_future.result(); result.has_value()) {
|
|
||||||
QFile::remove(m_output_path);
|
|
||||||
emitFailed(result.value());
|
|
||||||
} else {
|
|
||||||
emitSucceeded();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ExportToZipTask::abort()
|
|
||||||
{
|
|
||||||
if (m_build_zip_future.isRunning()) {
|
|
||||||
m_build_zip_future.cancel();
|
|
||||||
// NOTE: Here we don't do `emitAborted()` because it will be done when `m_build_zip_future` actually cancels, which may not occur
|
|
||||||
// immediately.
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
} // namespace MMCZip
|
|
||||||
|
@ -1,8 +1,7 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
* Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com>
|
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -36,163 +35,110 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <quazip.h>
|
|
||||||
#include <quazip/JlCompress.h>
|
|
||||||
#include <QDir>
|
|
||||||
#include <QFileInfo>
|
|
||||||
#include <QFuture>
|
|
||||||
#include <QFutureWatcher>
|
|
||||||
#include <QHash>
|
|
||||||
#include <QSet>
|
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
#include <QFileInfo>
|
||||||
|
#include <QSet>
|
||||||
|
#include "minecraft/mod/Mod.h"
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <memory>
|
|
||||||
|
#include <quazip/JlCompress.h>
|
||||||
#include <optional>
|
#include <optional>
|
||||||
|
|
||||||
#if defined(LAUNCHER_APPLICATION)
|
namespace MMCZip
|
||||||
#include "minecraft/mod/Mod.h"
|
{
|
||||||
#endif
|
using FilterFunction = std::function<bool(const QString &)>;
|
||||||
#include "tasks/Task.h"
|
|
||||||
|
|
||||||
namespace MMCZip {
|
/**
|
||||||
using FilterFunction = std::function<bool(const QString&)>;
|
* Merge two zip files, using a filter function
|
||||||
|
*/
|
||||||
|
bool mergeZipFiles(QuaZip *into, QFileInfo from, QSet<QString> &contained,
|
||||||
|
const FilterFunction filter = nullptr);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Merge two zip files, using a filter function
|
* Compress directory, by providing a list of files to compress
|
||||||
*/
|
* \param zip target archive
|
||||||
bool mergeZipFiles(QuaZip* into, QFileInfo from, QSet<QString>& contained, const FilterFunction filter = nullptr);
|
* \param dir directory that will be compressed (to compress with relative paths)
|
||||||
|
* \param files list of files to compress
|
||||||
|
* \param followSymlinks should follow symlinks when compressing file data
|
||||||
|
* \return true for success or false for failure
|
||||||
|
*/
|
||||||
|
bool compressDirFiles(QuaZip *zip, QString dir, QFileInfoList files, bool followSymlinks = false);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Compress directory, by providing a list of files to compress
|
* Compress directory, by providing a list of files to compress
|
||||||
* \param zip target archive
|
* \param fileCompressed target archive file
|
||||||
* \param dir directory that will be compressed (to compress with relative paths)
|
* \param dir directory that will be compressed (to compress with relative paths)
|
||||||
* \param files list of files to compress
|
* \param files list of files to compress
|
||||||
* \param followSymlinks should follow symlinks when compressing file data
|
* \param followSymlinks should follow symlinks when compressing file data
|
||||||
* \return true for success or false for failure
|
* \return true for success or false for failure
|
||||||
*/
|
*/
|
||||||
bool compressDirFiles(QuaZip* zip, QString dir, QFileInfoList files, bool followSymlinks = false);
|
bool compressDirFiles(QString fileCompressed, QString dir, QFileInfoList files, bool followSymlinks = false);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Compress directory, by providing a list of files to compress
|
* take a source jar, add mods to it, resulting in target jar
|
||||||
* \param fileCompressed target archive file
|
*/
|
||||||
* \param dir directory that will be compressed (to compress with relative paths)
|
bool createModdedJar(QString sourceJarPath, QString targetJarPath, const QList<Mod*>& mods);
|
||||||
* \param files list of files to compress
|
|
||||||
* \param followSymlinks should follow symlinks when compressing file data
|
|
||||||
* \return true for success or false for failure
|
|
||||||
*/
|
|
||||||
bool compressDirFiles(QString fileCompressed, QString dir, QFileInfoList files, bool followSymlinks = false);
|
|
||||||
|
|
||||||
#if defined(LAUNCHER_APPLICATION)
|
/**
|
||||||
/**
|
* Find a single file in archive by file name (not path)
|
||||||
* take a source jar, add mods to it, resulting in target jar
|
*
|
||||||
*/
|
* \param ignore_paths paths to skip when recursing the search
|
||||||
bool createModdedJar(QString sourceJarPath, QString targetJarPath, const QList<Mod*>& mods);
|
*
|
||||||
#endif
|
* \return the path prefix where the file is
|
||||||
/**
|
*/
|
||||||
* Find a single file in archive by file name (not path)
|
QString findFolderOfFileInZip(QuaZip * zip, const QString & what, const QStringList& ignore_paths = {}, const QString &root = QString(""));
|
||||||
*
|
|
||||||
* \param ignore_paths paths to skip when recursing the search
|
|
||||||
*
|
|
||||||
* \return the path prefix where the file is
|
|
||||||
*/
|
|
||||||
QString findFolderOfFileInZip(QuaZip* zip, const QString& what, const QStringList& ignore_paths = {}, const QString& root = QString(""));
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Find a multiple files of the same name in archive by file name
|
* Find a multiple files of the same name in archive by file name
|
||||||
* If a file is found in a path, no deeper paths are searched
|
* If a file is found in a path, no deeper paths are searched
|
||||||
*
|
*
|
||||||
* \return true if anything was found
|
* \return true if anything was found
|
||||||
*/
|
*/
|
||||||
bool findFilesInZip(QuaZip* zip, const QString& what, QStringList& result, const QString& root = QString());
|
bool findFilesInZip(QuaZip * zip, const QString & what, QStringList & result, const QString &root = QString());
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Extract a subdirectory from an archive
|
* Extract a subdirectory from an archive
|
||||||
*/
|
*/
|
||||||
std::optional<QStringList> extractSubDir(QuaZip* zip, const QString& subdir, const QString& target);
|
std::optional<QStringList> extractSubDir(QuaZip *zip, const QString & subdir, const QString &target);
|
||||||
|
|
||||||
bool extractRelFile(QuaZip* zip, const QString& file, const QString& target);
|
bool extractRelFile(QuaZip *zip, const QString & file, const QString &target);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Extract a whole archive.
|
* Extract a whole archive.
|
||||||
*
|
*
|
||||||
* \param fileCompressed The name of the archive.
|
* \param fileCompressed The name of the archive.
|
||||||
* \param dir The directory to extract to, the current directory if left empty.
|
* \param dir The directory to extract to, the current directory if left empty.
|
||||||
* \return The list of the full paths of the files extracted, empty on failure.
|
* \return The list of the full paths of the files extracted, empty on failure.
|
||||||
*/
|
*/
|
||||||
std::optional<QStringList> extractDir(QString fileCompressed, QString dir);
|
std::optional<QStringList> extractDir(QString fileCompressed, QString dir);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Extract a subdirectory from an archive
|
* Extract a subdirectory from an archive
|
||||||
*
|
*
|
||||||
* \param fileCompressed The name of the archive.
|
* \param fileCompressed The name of the archive.
|
||||||
* \param subdir The directory within the archive to extract
|
* \param subdir The directory within the archive to extract
|
||||||
* \param dir The directory to extract to, the current directory if left empty.
|
* \param dir The directory to extract to, the current directory if left empty.
|
||||||
* \return The list of the full paths of the files extracted, empty on failure.
|
* \return The list of the full paths of the files extracted, empty on failure.
|
||||||
*/
|
*/
|
||||||
std::optional<QStringList> extractDir(QString fileCompressed, QString subdir, QString dir);
|
std::optional<QStringList> extractDir(QString fileCompressed, QString subdir, QString dir);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Extract a single file from an archive into a directory
|
* Extract a single file from an archive into a directory
|
||||||
*
|
*
|
||||||
* \param fileCompressed The name of the archive.
|
* \param fileCompressed The name of the archive.
|
||||||
* \param file The file within the archive to extract
|
* \param file The file within the archive to extract
|
||||||
* \param dir The directory to extract to, the current directory if left empty.
|
* \param dir The directory to extract to, the current directory if left empty.
|
||||||
* \return true for success or false for failure
|
* \return true for success or false for failure
|
||||||
*/
|
*/
|
||||||
bool extractFile(QString fileCompressed, QString file, QString dir);
|
bool extractFile(QString fileCompressed, QString file, QString dir);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Populate a QFileInfoList with a directory tree recursively, while allowing to excludeFilter what shouldn't be included.
|
* Populate a QFileInfoList with a directory tree recursively, while allowing to excludeFilter what shouldn't be included.
|
||||||
* \param rootDir directory to start off
|
* \param rootDir directory to start off
|
||||||
* \param subDir subdirectory, should be nullptr for first invocation
|
* \param subDir subdirectory, should be nullptr for first invocation
|
||||||
* \param files resulting list of QFileInfo
|
* \param files resulting list of QFileInfo
|
||||||
* \param excludeFilter function to excludeFilter which files shouldn't be included (returning true means to excude)
|
* \param excludeFilter function to excludeFilter which files shouldn't be included (returning true means to excude)
|
||||||
* \return true for success or false for failure
|
* \return true for success or false for failure
|
||||||
*/
|
*/
|
||||||
bool collectFileListRecursively(const QString& rootDir, const QString& subDir, QFileInfoList* files, FilterFunction excludeFilter);
|
bool collectFileListRecursively(const QString &rootDir, const QString &subDir, QFileInfoList *files, FilterFunction excludeFilter);
|
||||||
|
}
|
||||||
#if defined(LAUNCHER_APPLICATION)
|
|
||||||
class ExportToZipTask : public Task {
|
|
||||||
public:
|
|
||||||
ExportToZipTask(QString outputPath, QDir dir, QFileInfoList files, QString destinationPrefix = "", bool followSymlinks = false)
|
|
||||||
: m_output_path(outputPath)
|
|
||||||
, m_output(outputPath)
|
|
||||||
, m_dir(dir)
|
|
||||||
, m_files(files)
|
|
||||||
, m_destination_prefix(destinationPrefix)
|
|
||||||
, m_follow_symlinks(followSymlinks)
|
|
||||||
{
|
|
||||||
setAbortable(true);
|
|
||||||
};
|
|
||||||
ExportToZipTask(QString outputPath, QString dir, QFileInfoList files, QString destinationPrefix = "", bool followSymlinks = false)
|
|
||||||
: ExportToZipTask(outputPath, QDir(dir), files, destinationPrefix, followSymlinks){};
|
|
||||||
|
|
||||||
virtual ~ExportToZipTask() = default;
|
|
||||||
|
|
||||||
void setExcludeFiles(QStringList excludeFiles) { m_exclude_files = excludeFiles; }
|
|
||||||
void addExtraFile(QString fileName, QByteArray data) { m_extra_files.insert(fileName, data); }
|
|
||||||
|
|
||||||
using ZipResult = std::optional<QString>;
|
|
||||||
|
|
||||||
protected:
|
|
||||||
virtual void executeTask() override;
|
|
||||||
bool abort() override;
|
|
||||||
|
|
||||||
ZipResult exportZip();
|
|
||||||
void finish();
|
|
||||||
|
|
||||||
private:
|
|
||||||
QString m_output_path;
|
|
||||||
QuaZip m_output;
|
|
||||||
QDir m_dir;
|
|
||||||
QFileInfoList m_files;
|
|
||||||
QString m_destination_prefix;
|
|
||||||
bool m_follow_symlinks;
|
|
||||||
QStringList m_exclude_files;
|
|
||||||
QHash<QString, QByteArray> m_extra_files;
|
|
||||||
|
|
||||||
QFuture<ZipResult> m_build_zip_future;
|
|
||||||
QFutureWatcher<ZipResult> m_build_zip_watcher;
|
|
||||||
};
|
|
||||||
#endif
|
|
||||||
} // namespace MMCZip
|
|
||||||
|
@ -1,11 +1,8 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QCoreApplication>
|
#include <QCoreApplication>
|
||||||
#include <QDebug>
|
|
||||||
#include <QPixmapCache>
|
#include <QPixmapCache>
|
||||||
#include <QThread>
|
#include <QThread>
|
||||||
#include <QTime>
|
|
||||||
#include <limits>
|
|
||||||
|
|
||||||
#define GET_TYPE() \
|
#define GET_TYPE() \
|
||||||
Qt::ConnectionType type; \
|
Qt::ConnectionType type; \
|
||||||
@ -63,8 +60,6 @@ class PixmapCache final : public QObject {
|
|||||||
DEFINE_FUNC_ONE_PARAM(remove, bool, const QPixmapCache::Key&)
|
DEFINE_FUNC_ONE_PARAM(remove, bool, const QPixmapCache::Key&)
|
||||||
DEFINE_FUNC_TWO_PARAM(replace, bool, const QPixmapCache::Key&, const QPixmap&)
|
DEFINE_FUNC_TWO_PARAM(replace, bool, const QPixmapCache::Key&, const QPixmap&)
|
||||||
DEFINE_FUNC_ONE_PARAM(setCacheLimit, bool, int)
|
DEFINE_FUNC_ONE_PARAM(setCacheLimit, bool, int)
|
||||||
DEFINE_FUNC_NO_PARAM(markCacheMissByEviciton, bool)
|
|
||||||
DEFINE_FUNC_ONE_PARAM(setFastEvictionThreshold, bool, int)
|
|
||||||
|
|
||||||
// NOTE: Every function returns something non-void to simplify the macros.
|
// NOTE: Every function returns something non-void to simplify the macros.
|
||||||
private slots:
|
private slots:
|
||||||
@ -95,53 +90,6 @@ class PixmapCache final : public QObject {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Mark that a cache miss occurred because of a eviction if too many of these occur too fast the cache size is increased
|
|
||||||
* @return if the cache size was increased
|
|
||||||
*/
|
|
||||||
bool _markCacheMissByEviciton()
|
|
||||||
{
|
|
||||||
static constexpr uint maxInt = static_cast<uint>(std::numeric_limits<int>::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 < oneSecond) { // less than a second ago
|
|
||||||
++m_consecutive_fast_evicitons;
|
|
||||||
} else {
|
|
||||||
m_consecutive_fast_evicitons = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
m_last_cache_miss_by_eviciton = now;
|
|
||||||
if (m_consecutive_fast_evicitons >= m_consecutive_fast_evicitons_threshold) {
|
|
||||||
// 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<int>(newSize);
|
|
||||||
}
|
|
||||||
_setCacheLimit(static_cast<int>(newSize));
|
|
||||||
m_consecutive_fast_evicitons = 0;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool _setFastEvictionThreshold(int threshold)
|
|
||||||
{
|
|
||||||
m_consecutive_fast_evicitons_threshold = threshold;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static PixmapCache* s_instance;
|
static PixmapCache* s_instance;
|
||||||
QTime m_last_cache_miss_by_eviciton;
|
|
||||||
int m_consecutive_fast_evicitons = 0;
|
|
||||||
int m_consecutive_fast_evicitons_threshold = 15;
|
|
||||||
};
|
};
|
||||||
|
@ -16,25 +16,15 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <QDebug>
|
#include <QStringList>
|
||||||
#include <QDir>
|
#include <QDir>
|
||||||
#include <QString>
|
#include <QString>
|
||||||
#include <QStringList>
|
|
||||||
#include <QSysInfo>
|
#include <QSysInfo>
|
||||||
#include <QtGlobal>
|
#include <QtGlobal>
|
||||||
|
|
||||||
|
#include "MangoHud.h"
|
||||||
#include "FileSystem.h"
|
#include "FileSystem.h"
|
||||||
#include "Json.h"
|
#include "Json.h"
|
||||||
#include "MangoHud.h"
|
|
||||||
|
|
||||||
#ifdef __GLIBC__
|
|
||||||
#ifndef _GNU_SOURCE
|
|
||||||
#define _GNU_SOURCE
|
|
||||||
#define UNDEF_GNU_SOURCE
|
|
||||||
#endif
|
|
||||||
#include <dlfcn.h>
|
|
||||||
#include <linux/limits.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace MangoHud {
|
namespace MangoHud {
|
||||||
|
|
||||||
@ -116,37 +106,4 @@ QString getLibraryString()
|
|||||||
|
|
||||||
return QString();
|
return QString();
|
||||||
}
|
}
|
||||||
|
|
||||||
QString findLibrary(QString libName)
|
|
||||||
{
|
|
||||||
#ifdef __GLIBC__
|
|
||||||
const char* library = libName.toLocal8Bit().constData();
|
|
||||||
|
|
||||||
void* handle = dlopen(library, RTLD_NOW);
|
|
||||||
if (!handle) {
|
|
||||||
qCritical() << "dlopen() failed:" << dlerror();
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
char path[PATH_MAX];
|
|
||||||
if (dlinfo(handle, RTLD_DI_ORIGIN, path) == -1) {
|
|
||||||
qCritical() << "dlinfo() failed:" << dlerror();
|
|
||||||
dlclose(handle);
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
auto fullPath = FS::PathCombine(QString(path), libName);
|
|
||||||
|
|
||||||
dlclose(handle);
|
|
||||||
return fullPath;
|
|
||||||
#else
|
|
||||||
qWarning() << "MangoHud::findLibrary is not implemented on this platform";
|
|
||||||
return {};
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
} // namespace MangoHud
|
} // namespace MangoHud
|
||||||
|
|
||||||
#ifdef UNDEF_GNU_SOURCE
|
|
||||||
#undef _GNU_SOURCE
|
|
||||||
#undef UNDEF_GNU_SOURCE
|
|
||||||
#endif
|
|
||||||
|
@ -24,6 +24,4 @@
|
|||||||
namespace MangoHud {
|
namespace MangoHud {
|
||||||
|
|
||||||
QString getLibraryString();
|
QString getLibraryString();
|
||||||
|
}
|
||||||
QString findLibrary(QString libName);
|
|
||||||
} // namespace MangoHud
|
|
||||||
|
@ -18,7 +18,7 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <cmark.h>
|
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
#include <cmark.h>
|
||||||
|
|
||||||
QString markdownToHTML(const QString& markdown);
|
QString markdownToHTML(const QString& markdown);
|
@ -22,11 +22,12 @@ MessageLevel::Enum MessageLevel::getLevel(const QString& levelName)
|
|||||||
return MessageLevel::Unknown;
|
return MessageLevel::Unknown;
|
||||||
}
|
}
|
||||||
|
|
||||||
MessageLevel::Enum MessageLevel::fromLine(QString& line)
|
MessageLevel::Enum MessageLevel::fromLine(QString &line)
|
||||||
{
|
{
|
||||||
// Level prefix
|
// Level prefix
|
||||||
int endmark = line.indexOf("]!");
|
int endmark = line.indexOf("]!");
|
||||||
if (line.startsWith("!![") && endmark != -1) {
|
if (line.startsWith("!![") && endmark != -1)
|
||||||
|
{
|
||||||
auto level = MessageLevel::getLevel(line.left(endmark).mid(3));
|
auto level = MessageLevel::getLevel(line.left(endmark).mid(3));
|
||||||
line = line.mid(endmark + 2);
|
line = line.mid(endmark + 2);
|
||||||
return level;
|
return level;
|
||||||
|
@ -6,21 +6,23 @@
|
|||||||
* @brief the MessageLevel Enum
|
* @brief the MessageLevel Enum
|
||||||
* defines what level a log message is
|
* defines what level a log message is
|
||||||
*/
|
*/
|
||||||
namespace MessageLevel {
|
namespace MessageLevel
|
||||||
enum Enum {
|
{
|
||||||
Unknown, /**< No idea what this is or where it came from */
|
enum Enum
|
||||||
StdOut, /**< Undetermined stderr messages */
|
{
|
||||||
StdErr, /**< Undetermined stdout messages */
|
Unknown, /**< No idea what this is or where it came from */
|
||||||
|
StdOut, /**< Undetermined stderr messages */
|
||||||
|
StdErr, /**< Undetermined stdout messages */
|
||||||
Launcher, /**< Launcher Messages */
|
Launcher, /**< Launcher Messages */
|
||||||
Debug, /**< Debug Messages */
|
Debug, /**< Debug Messages */
|
||||||
Info, /**< Info Messages */
|
Info, /**< Info Messages */
|
||||||
Message, /**< Standard Messages */
|
Message, /**< Standard Messages */
|
||||||
Warning, /**< Warnings */
|
Warning, /**< Warnings */
|
||||||
Error, /**< Errors */
|
Error, /**< Errors */
|
||||||
Fatal, /**< Fatal Errors */
|
Fatal, /**< Fatal Errors */
|
||||||
};
|
};
|
||||||
MessageLevel::Enum getLevel(const QString& levelName);
|
MessageLevel::Enum getLevel(const QString &levelName);
|
||||||
|
|
||||||
/* Get message level from a line. Line is modified if it was successful. */
|
/* Get message level from a line. Line is modified if it was successful. */
|
||||||
MessageLevel::Enum fromLine(QString& line);
|
MessageLevel::Enum fromLine(QString &line);
|
||||||
} // namespace MessageLevel
|
}
|
||||||
|
@ -1,8 +1,7 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
* Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me>
|
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -38,39 +37,88 @@
|
|||||||
#include "BaseInstance.h"
|
#include "BaseInstance.h"
|
||||||
#include "launch/LaunchTask.h"
|
#include "launch/LaunchTask.h"
|
||||||
|
|
||||||
class NullInstance : public BaseInstance {
|
class NullInstance: public BaseInstance
|
||||||
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
NullInstance(SettingsObjectPtr globalSettings, SettingsObjectPtr settings, const QString& rootDir)
|
NullInstance(SettingsObjectPtr globalSettings, SettingsObjectPtr settings, const QString& rootDir)
|
||||||
: BaseInstance(globalSettings, settings, rootDir)
|
:BaseInstance(globalSettings, settings, rootDir)
|
||||||
{
|
{
|
||||||
setVersionBroken(true);
|
setVersionBroken(true);
|
||||||
}
|
}
|
||||||
virtual ~NullInstance(){};
|
virtual ~NullInstance() {};
|
||||||
void saveNow() override {}
|
void saveNow() override
|
||||||
void loadSpecificSettings() override { setSpecificSettingsLoaded(true); }
|
{
|
||||||
QString getStatusbarDescription() override { return tr("Unknown instance type"); };
|
}
|
||||||
QSet<QString> traits() const override { return {}; };
|
void loadSpecificSettings() override
|
||||||
QString instanceConfigFolder() const override { return instanceRoot(); };
|
{
|
||||||
shared_qobject_ptr<LaunchTask> createLaunchTask(AuthSessionPtr, MinecraftServerTargetPtr) override { return nullptr; }
|
setSpecificSettingsLoaded(true);
|
||||||
shared_qobject_ptr<Task> createUpdateTask([[maybe_unused]] Net::Mode mode) override { return nullptr; }
|
}
|
||||||
QProcessEnvironment createEnvironment() override { return QProcessEnvironment(); }
|
QString getStatusbarDescription() override
|
||||||
QProcessEnvironment createLaunchEnvironment() override { return QProcessEnvironment(); }
|
{
|
||||||
QMap<QString, QString> getVariables() override { return QMap<QString, QString>(); }
|
return tr("Unknown instance type");
|
||||||
IPathMatcher::Ptr getLogFileMatcher() override { return nullptr; }
|
};
|
||||||
QString getLogFileRoot() override { return instanceRoot(); }
|
QSet< QString > traits() const override
|
||||||
QString typeName() const override { return "Null"; }
|
{
|
||||||
bool canExport() const override { return false; }
|
return {};
|
||||||
bool canEdit() const override { return false; }
|
};
|
||||||
bool canLaunch() const override { return false; }
|
QString instanceConfigFolder() const override
|
||||||
void populateLaunchMenu(QMenu* menu) override {}
|
{
|
||||||
|
return instanceRoot();
|
||||||
|
};
|
||||||
|
shared_qobject_ptr<LaunchTask> createLaunchTask(AuthSessionPtr, MinecraftServerTargetPtr) override
|
||||||
|
{
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
shared_qobject_ptr< Task > createUpdateTask(Net::Mode mode) override
|
||||||
|
{
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
QProcessEnvironment createEnvironment() override
|
||||||
|
{
|
||||||
|
return QProcessEnvironment();
|
||||||
|
}
|
||||||
|
QProcessEnvironment createLaunchEnvironment() override
|
||||||
|
{
|
||||||
|
return QProcessEnvironment();
|
||||||
|
}
|
||||||
|
QMap<QString, QString> getVariables() override
|
||||||
|
{
|
||||||
|
return QMap<QString, QString>();
|
||||||
|
}
|
||||||
|
IPathMatcher::Ptr getLogFileMatcher() override
|
||||||
|
{
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
QString getLogFileRoot() override
|
||||||
|
{
|
||||||
|
return instanceRoot();
|
||||||
|
}
|
||||||
|
QString typeName() const override
|
||||||
|
{
|
||||||
|
return "Null";
|
||||||
|
}
|
||||||
|
bool canExport() const override
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
bool canEdit() const override
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
bool canLaunch() const override
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
QStringList verboseDescription(AuthSessionPtr session, MinecraftServerTargetPtr serverToJoin) override
|
QStringList verboseDescription(AuthSessionPtr session, MinecraftServerTargetPtr serverToJoin) override
|
||||||
{
|
{
|
||||||
QStringList out;
|
QStringList out;
|
||||||
out << "Null instance - placeholder.";
|
out << "Null instance - placeholder.";
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
QString modsRoot() const override { return QString(); }
|
QString modsRoot() const override {
|
||||||
|
return QString();
|
||||||
|
}
|
||||||
void updateRuntimeContext()
|
void updateRuntimeContext()
|
||||||
{
|
{
|
||||||
// NOOP
|
// NOOP
|
||||||
|
@ -1,35 +1,47 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QList>
|
enum class ProblemSeverity
|
||||||
#include <QString>
|
{
|
||||||
|
None,
|
||||||
|
Warning,
|
||||||
|
Error
|
||||||
|
};
|
||||||
|
|
||||||
enum class ProblemSeverity { None, Warning, Error };
|
struct PatchProblem
|
||||||
|
{
|
||||||
struct PatchProblem {
|
|
||||||
ProblemSeverity m_severity;
|
ProblemSeverity m_severity;
|
||||||
QString m_description;
|
QString m_description;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ProblemProvider {
|
class ProblemProvider
|
||||||
public:
|
{
|
||||||
virtual ~ProblemProvider() {}
|
public:
|
||||||
|
virtual ~ProblemProvider() {};
|
||||||
virtual const QList<PatchProblem> getProblems() const = 0;
|
virtual const QList<PatchProblem> getProblems() const = 0;
|
||||||
virtual ProblemSeverity getProblemSeverity() const = 0;
|
virtual ProblemSeverity getProblemSeverity() const = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ProblemContainer : public ProblemProvider {
|
class ProblemContainer : public ProblemProvider
|
||||||
public:
|
{
|
||||||
const QList<PatchProblem> getProblems() const override { return m_problems; }
|
public:
|
||||||
ProblemSeverity getProblemSeverity() const override { return m_problemSeverity; }
|
const QList<PatchProblem> getProblems() const override
|
||||||
virtual void addProblem(ProblemSeverity severity, const QString& description)
|
|
||||||
{
|
{
|
||||||
if (severity > m_problemSeverity) {
|
return m_problems;
|
||||||
|
}
|
||||||
|
ProblemSeverity getProblemSeverity() const override
|
||||||
|
{
|
||||||
|
return m_problemSeverity;
|
||||||
|
}
|
||||||
|
virtual void addProblem(ProblemSeverity severity, const QString &description)
|
||||||
|
{
|
||||||
|
if(severity > m_problemSeverity)
|
||||||
|
{
|
||||||
m_problemSeverity = severity;
|
m_problemSeverity = severity;
|
||||||
}
|
}
|
||||||
m_problems.append({ severity, description });
|
m_problems.append({severity, description});
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QList<PatchProblem> m_problems;
|
QList<PatchProblem> m_problems;
|
||||||
ProblemSeverity m_problemSeverity = ProblemSeverity::None;
|
ProblemSeverity m_problemSeverity = ProblemSeverity::None;
|
||||||
};
|
};
|
||||||
|
@ -36,34 +36,35 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QList>
|
|
||||||
#include <QVariant>
|
#include <QVariant>
|
||||||
|
#include <QList>
|
||||||
|
|
||||||
namespace QVariantUtils {
|
namespace QVariantUtils {
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
inline QList<T> toList(QVariant src)
|
inline QList<T> toList(QVariant src) {
|
||||||
{
|
|
||||||
QVariantList variantList = src.toList();
|
QVariantList variantList = src.toList();
|
||||||
|
|
||||||
QList<T> list_t;
|
QList<T> list_t;
|
||||||
list_t.reserve(variantList.size());
|
list_t.reserve(variantList.size());
|
||||||
for (const QVariant& v : variantList) {
|
for (const QVariant& v : variantList)
|
||||||
|
{
|
||||||
list_t.append(v.value<T>());
|
list_t.append(v.value<T>());
|
||||||
}
|
}
|
||||||
return list_t;
|
return list_t;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
inline QVariant fromList(QList<T> val)
|
inline QVariant fromList(QList<T> val) {
|
||||||
{
|
|
||||||
QVariantList variantList;
|
QVariantList variantList;
|
||||||
variantList.reserve(val.size());
|
variantList.reserve(val.size());
|
||||||
for (const T& v : val) {
|
for (const T& v : val)
|
||||||
|
{
|
||||||
variantList.append(v);
|
variantList.append(v);
|
||||||
}
|
}
|
||||||
|
|
||||||
return variantList;
|
return variantList;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace QVariantUtils
|
}
|
@ -1,12 +1,13 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <QMap>
|
|
||||||
#include <QReadLocker>
|
|
||||||
#include <QSet>
|
|
||||||
#include <QWriteLocker>
|
#include <QWriteLocker>
|
||||||
|
#include <QReadLocker>
|
||||||
|
#include <QMap>
|
||||||
|
#include <QSet>
|
||||||
|
|
||||||
template <typename K, typename V>
|
template <typename K, typename V>
|
||||||
class RWStorage {
|
class RWStorage
|
||||||
public:
|
{
|
||||||
|
public:
|
||||||
void add(K key, V value)
|
void add(K key, V value)
|
||||||
{
|
{
|
||||||
QWriteLocker l(&lock);
|
QWriteLocker l(&lock);
|
||||||
@ -16,19 +17,21 @@ class RWStorage {
|
|||||||
V get(K key)
|
V get(K key)
|
||||||
{
|
{
|
||||||
QReadLocker l(&lock);
|
QReadLocker l(&lock);
|
||||||
if (cache.contains(key)) {
|
if(cache.contains(key))
|
||||||
|
{
|
||||||
return cache[key];
|
return cache[key];
|
||||||
} else
|
}
|
||||||
return V();
|
else return V();
|
||||||
}
|
}
|
||||||
bool get(K key, V& value)
|
bool get(K key, V& value)
|
||||||
{
|
{
|
||||||
QReadLocker l(&lock);
|
QReadLocker l(&lock);
|
||||||
if (cache.contains(key)) {
|
if(cache.contains(key))
|
||||||
|
{
|
||||||
value = cache[key];
|
value = cache[key];
|
||||||
return true;
|
return true;
|
||||||
} else
|
}
|
||||||
return false;
|
else return false;
|
||||||
}
|
}
|
||||||
bool has(K key)
|
bool has(K key)
|
||||||
{
|
{
|
||||||
@ -38,14 +41,15 @@ class RWStorage {
|
|||||||
bool stale(K key)
|
bool stale(K key)
|
||||||
{
|
{
|
||||||
QReadLocker l(&lock);
|
QReadLocker l(&lock);
|
||||||
if (!cache.contains(key))
|
if(!cache.contains(key))
|
||||||
return true;
|
return true;
|
||||||
return stale_entries.contains(key);
|
return stale_entries.contains(key);
|
||||||
}
|
}
|
||||||
void setStale(K key)
|
void setStale(K key)
|
||||||
{
|
{
|
||||||
QWriteLocker l(&lock);
|
QWriteLocker l(&lock);
|
||||||
if (cache.contains(key)) {
|
if(cache.contains(key))
|
||||||
|
{
|
||||||
stale_entries.insert(key);
|
stale_entries.insert(key);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -55,8 +59,7 @@ class RWStorage {
|
|||||||
cache.clear();
|
cache.clear();
|
||||||
stale_entries.clear();
|
stale_entries.clear();
|
||||||
}
|
}
|
||||||
|
private:
|
||||||
private:
|
|
||||||
QReadWriteLock lock;
|
QReadWriteLock lock;
|
||||||
QMap<K, V> cache;
|
QMap<K, V> cache;
|
||||||
QSet<K> stale_entries;
|
QSet<K> stale_entries;
|
||||||
|
@ -1,21 +1,25 @@
|
|||||||
#include "RecursiveFileSystemWatcher.h"
|
#include "RecursiveFileSystemWatcher.h"
|
||||||
|
|
||||||
#include <QDebug>
|
|
||||||
#include <QRegularExpression>
|
#include <QRegularExpression>
|
||||||
|
#include <QDebug>
|
||||||
|
|
||||||
RecursiveFileSystemWatcher::RecursiveFileSystemWatcher(QObject* parent) : QObject(parent), m_watcher(new QFileSystemWatcher(this))
|
RecursiveFileSystemWatcher::RecursiveFileSystemWatcher(QObject *parent)
|
||||||
|
: QObject(parent), m_watcher(new QFileSystemWatcher(this))
|
||||||
{
|
{
|
||||||
connect(m_watcher, &QFileSystemWatcher::fileChanged, this, &RecursiveFileSystemWatcher::fileChange);
|
connect(m_watcher, &QFileSystemWatcher::fileChanged, this,
|
||||||
connect(m_watcher, &QFileSystemWatcher::directoryChanged, this, &RecursiveFileSystemWatcher::directoryChange);
|
&RecursiveFileSystemWatcher::fileChange);
|
||||||
|
connect(m_watcher, &QFileSystemWatcher::directoryChanged, this,
|
||||||
|
&RecursiveFileSystemWatcher::directoryChange);
|
||||||
}
|
}
|
||||||
|
|
||||||
void RecursiveFileSystemWatcher::setRootDir(const QDir& root)
|
void RecursiveFileSystemWatcher::setRootDir(const QDir &root)
|
||||||
{
|
{
|
||||||
bool wasEnabled = m_isEnabled;
|
bool wasEnabled = m_isEnabled;
|
||||||
disable();
|
disable();
|
||||||
m_root = root;
|
m_root = root;
|
||||||
setFiles(scanRecursive(m_root));
|
setFiles(scanRecursive(m_root));
|
||||||
if (wasEnabled) {
|
if (wasEnabled)
|
||||||
|
{
|
||||||
enable();
|
enable();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -24,14 +28,16 @@ void RecursiveFileSystemWatcher::setWatchFiles(const bool watchFiles)
|
|||||||
bool wasEnabled = m_isEnabled;
|
bool wasEnabled = m_isEnabled;
|
||||||
disable();
|
disable();
|
||||||
m_watchFiles = watchFiles;
|
m_watchFiles = watchFiles;
|
||||||
if (wasEnabled) {
|
if (wasEnabled)
|
||||||
|
{
|
||||||
enable();
|
enable();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void RecursiveFileSystemWatcher::enable()
|
void RecursiveFileSystemWatcher::enable()
|
||||||
{
|
{
|
||||||
if (m_isEnabled) {
|
if (m_isEnabled)
|
||||||
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
Q_ASSERT(m_root != QDir::root());
|
Q_ASSERT(m_root != QDir::root());
|
||||||
@ -40,7 +46,8 @@ void RecursiveFileSystemWatcher::enable()
|
|||||||
}
|
}
|
||||||
void RecursiveFileSystemWatcher::disable()
|
void RecursiveFileSystemWatcher::disable()
|
||||||
{
|
{
|
||||||
if (!m_isEnabled) {
|
if (!m_isEnabled)
|
||||||
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
m_isEnabled = false;
|
m_isEnabled = false;
|
||||||
@ -48,49 +55,57 @@ void RecursiveFileSystemWatcher::disable()
|
|||||||
m_watcher->removePaths(m_watcher->directories());
|
m_watcher->removePaths(m_watcher->directories());
|
||||||
}
|
}
|
||||||
|
|
||||||
void RecursiveFileSystemWatcher::setFiles(const QStringList& files)
|
void RecursiveFileSystemWatcher::setFiles(const QStringList &files)
|
||||||
{
|
{
|
||||||
if (files != m_files) {
|
if (files != m_files)
|
||||||
|
{
|
||||||
m_files = files;
|
m_files = files;
|
||||||
emit filesChanged();
|
emit filesChanged();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void RecursiveFileSystemWatcher::addFilesToWatcherRecursive(const QDir& dir)
|
void RecursiveFileSystemWatcher::addFilesToWatcherRecursive(const QDir &dir)
|
||||||
{
|
{
|
||||||
m_watcher->addPath(dir.absolutePath());
|
m_watcher->addPath(dir.absolutePath());
|
||||||
for (const QString& directory : dir.entryList(QDir::Dirs | QDir::NoDotAndDotDot)) {
|
for (const QString &directory : dir.entryList(QDir::Dirs | QDir::NoDotAndDotDot))
|
||||||
|
{
|
||||||
addFilesToWatcherRecursive(dir.absoluteFilePath(directory));
|
addFilesToWatcherRecursive(dir.absoluteFilePath(directory));
|
||||||
}
|
}
|
||||||
if (m_watchFiles) {
|
if (m_watchFiles)
|
||||||
for (const QFileInfo& info : dir.entryInfoList(QDir::Files)) {
|
{
|
||||||
|
for (const QFileInfo &info : dir.entryInfoList(QDir::Files))
|
||||||
|
{
|
||||||
m_watcher->addPath(info.absoluteFilePath());
|
m_watcher->addPath(info.absoluteFilePath());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
QStringList RecursiveFileSystemWatcher::scanRecursive(const QDir& directory)
|
QStringList RecursiveFileSystemWatcher::scanRecursive(const QDir &directory)
|
||||||
{
|
{
|
||||||
QStringList ret;
|
QStringList ret;
|
||||||
if (!m_matcher) {
|
if(!m_matcher)
|
||||||
|
{
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
for (const QString& dir : directory.entryList(QDir::Dirs | QDir::NoDotAndDotDot | QDir::Hidden)) {
|
for (const QString &dir : directory.entryList(QDir::Dirs | QDir::NoDotAndDotDot | QDir::Hidden))
|
||||||
|
{
|
||||||
ret.append(scanRecursive(directory.absoluteFilePath(dir)));
|
ret.append(scanRecursive(directory.absoluteFilePath(dir)));
|
||||||
}
|
}
|
||||||
for (const QString& file : directory.entryList(QDir::Files | QDir::Hidden)) {
|
for (const QString &file : directory.entryList(QDir::Files | QDir::Hidden))
|
||||||
|
{
|
||||||
auto relPath = m_root.relativeFilePath(directory.absoluteFilePath(file));
|
auto relPath = m_root.relativeFilePath(directory.absoluteFilePath(file));
|
||||||
if (m_matcher->matches(relPath)) {
|
if (m_matcher->matches(relPath))
|
||||||
|
{
|
||||||
ret.append(relPath);
|
ret.append(relPath);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
void RecursiveFileSystemWatcher::fileChange(const QString& path)
|
void RecursiveFileSystemWatcher::fileChange(const QString &path)
|
||||||
{
|
{
|
||||||
emit fileChanged(path);
|
emit fileChanged(path);
|
||||||
}
|
}
|
||||||
void RecursiveFileSystemWatcher::directoryChange([[maybe_unused]] const QString& path)
|
void RecursiveFileSystemWatcher::directoryChange(const QString &path)
|
||||||
{
|
{
|
||||||
setFiles(scanRecursive(m_root));
|
setFiles(scanRecursive(m_root));
|
||||||
}
|
}
|
||||||
|
@ -1,48 +1,61 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QDir>
|
|
||||||
#include <QFileSystemWatcher>
|
#include <QFileSystemWatcher>
|
||||||
|
#include <QDir>
|
||||||
#include "pathmatcher/IPathMatcher.h"
|
#include "pathmatcher/IPathMatcher.h"
|
||||||
|
|
||||||
class RecursiveFileSystemWatcher : public QObject {
|
class RecursiveFileSystemWatcher : public QObject
|
||||||
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
RecursiveFileSystemWatcher(QObject* parent);
|
RecursiveFileSystemWatcher(QObject *parent);
|
||||||
|
|
||||||
void setRootDir(const QDir& root);
|
void setRootDir(const QDir &root);
|
||||||
QDir rootDir() const { return m_root; }
|
QDir rootDir() const
|
||||||
|
{
|
||||||
|
return m_root;
|
||||||
|
}
|
||||||
|
|
||||||
// WARNING: setting this to true may be bad for performance
|
// WARNING: setting this to true may be bad for performance
|
||||||
void setWatchFiles(const bool watchFiles);
|
void setWatchFiles(const bool watchFiles);
|
||||||
bool watchFiles() const { return m_watchFiles; }
|
bool watchFiles() const
|
||||||
|
{
|
||||||
|
return m_watchFiles;
|
||||||
|
}
|
||||||
|
|
||||||
void setMatcher(IPathMatcher::Ptr matcher) { m_matcher = matcher; }
|
void setMatcher(IPathMatcher::Ptr matcher)
|
||||||
|
{
|
||||||
|
m_matcher = matcher;
|
||||||
|
}
|
||||||
|
|
||||||
QStringList files() const { return m_files; }
|
QStringList files() const
|
||||||
|
{
|
||||||
|
return m_files;
|
||||||
|
}
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void filesChanged();
|
void filesChanged();
|
||||||
void fileChanged(const QString& path);
|
void fileChanged(const QString &path);
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void enable();
|
void enable();
|
||||||
void disable();
|
void disable();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QDir m_root;
|
QDir m_root;
|
||||||
bool m_watchFiles = false;
|
bool m_watchFiles = false;
|
||||||
bool m_isEnabled = false;
|
bool m_isEnabled = false;
|
||||||
IPathMatcher::Ptr m_matcher;
|
IPathMatcher::Ptr m_matcher;
|
||||||
|
|
||||||
QFileSystemWatcher* m_watcher;
|
QFileSystemWatcher *m_watcher;
|
||||||
|
|
||||||
QStringList m_files;
|
QStringList m_files;
|
||||||
void setFiles(const QStringList& files);
|
void setFiles(const QStringList &files);
|
||||||
|
|
||||||
void addFilesToWatcherRecursive(const QDir& dir);
|
void addFilesToWatcherRecursive(const QDir &dir);
|
||||||
QStringList scanRecursive(const QDir& dir);
|
QStringList scanRecursive(const QDir &dir);
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void fileChange(const QString& path);
|
void fileChange(const QString &path);
|
||||||
void directoryChange(const QString& path);
|
void directoryChange(const QString &path);
|
||||||
};
|
};
|
||||||
|
@ -24,8 +24,6 @@
|
|||||||
#include "minecraft/mod/ModFolderModel.h"
|
#include "minecraft/mod/ModFolderModel.h"
|
||||||
#include "minecraft/mod/ResourceFolderModel.h"
|
#include "minecraft/mod/ResourceFolderModel.h"
|
||||||
|
|
||||||
#include "net/ApiDownload.h"
|
|
||||||
|
|
||||||
ResourceDownloadTask::ResourceDownloadTask(ModPlatform::IndexedPack::Ptr pack,
|
ResourceDownloadTask::ResourceDownloadTask(ModPlatform::IndexedPack::Ptr pack,
|
||||||
ModPlatform::IndexedVersion version,
|
ModPlatform::IndexedVersion version,
|
||||||
const std::shared_ptr<ResourceFolderModel> packs,
|
const std::shared_ptr<ResourceFolderModel> packs,
|
||||||
@ -53,10 +51,10 @@ ResourceDownloadTask::ResourceDownloadTask(ModPlatform::IndexedPack::Ptr pack,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
m_filesNetJob->addNetAction(Net::ApiDownload::makeFile(m_pack_version.downloadUrl, dir.absoluteFilePath(getFilename())));
|
m_filesNetJob->addNetAction(Net::Download::makeFile(m_pack_version.downloadUrl, dir.absoluteFilePath(getFilename())));
|
||||||
connect(m_filesNetJob.get(), &NetJob::succeeded, this, &ResourceDownloadTask::downloadSucceeded);
|
connect(m_filesNetJob.get(), &NetJob::succeeded, this, &ResourceDownloadTask::downloadSucceeded);
|
||||||
connect(m_filesNetJob.get(), &NetJob::progress, this, &ResourceDownloadTask::downloadProgressChanged);
|
connect(m_filesNetJob.get(), &NetJob::progress, this, &ResourceDownloadTask::downloadProgressChanged);
|
||||||
connect(m_filesNetJob.get(), &NetJob::stepProgress, this, &ResourceDownloadTask::propagateStepProgress);
|
connect(m_filesNetJob.get(), &NetJob::stepProgress, this, &ResourceDownloadTask::propogateStepProgress);
|
||||||
connect(m_filesNetJob.get(), &NetJob::failed, this, &ResourceDownloadTask::downloadFailed);
|
connect(m_filesNetJob.get(), &NetJob::failed, this, &ResourceDownloadTask::downloadFailed);
|
||||||
|
|
||||||
addTask(m_filesNetJob);
|
addTask(m_filesNetJob);
|
||||||
|
@ -38,8 +38,6 @@ class ResourceDownloadTask : public SequentialTask {
|
|||||||
const QString& getFilename() const { return m_pack_version.fileName; }
|
const QString& getFilename() const { return m_pack_version.fileName; }
|
||||||
const QString& getCustomPath() const { return m_custom_target_folder; }
|
const QString& getCustomPath() const { return m_custom_target_folder; }
|
||||||
const QVariant& getVersionID() const { return m_pack_version.fileId; }
|
const QVariant& getVersionID() const { return m_pack_version.fileId; }
|
||||||
const ModPlatform::IndexedVersion& getVersion() const { return m_pack_version; }
|
|
||||||
const ModPlatform::ResourceProvider& getProvider() const { return m_pack->provider; }
|
|
||||||
const QString& getName() const { return m_pack->name; }
|
const QString& getName() const { return m_pack->name; }
|
||||||
ModPlatform::IndexedPack::Ptr getPack() { return m_pack; }
|
ModPlatform::IndexedPack::Ptr getPack() { return m_pack; }
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
@ -1,32 +1,44 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <QMap>
|
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
#include <QMap>
|
||||||
#include <QStringList>
|
#include <QStringList>
|
||||||
|
|
||||||
template <char Tseparator>
|
template <char Tseparator>
|
||||||
class SeparatorPrefixTree {
|
class SeparatorPrefixTree
|
||||||
public:
|
{
|
||||||
SeparatorPrefixTree(QStringList paths) { insert(paths); }
|
public:
|
||||||
|
SeparatorPrefixTree(QStringList paths)
|
||||||
|
{
|
||||||
|
insert(paths);
|
||||||
|
}
|
||||||
|
|
||||||
SeparatorPrefixTree(bool contained = false) { m_contained = contained; }
|
SeparatorPrefixTree(bool contained = false)
|
||||||
|
{
|
||||||
|
m_contained = contained;
|
||||||
|
}
|
||||||
|
|
||||||
void insert(QStringList paths)
|
void insert(QStringList paths)
|
||||||
{
|
{
|
||||||
for (auto& path : paths) {
|
for(auto &path: paths)
|
||||||
|
{
|
||||||
insert(path);
|
insert(path);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// insert an exact path into the tree
|
/// insert an exact path into the tree
|
||||||
SeparatorPrefixTree& insert(QString path)
|
SeparatorPrefixTree & insert(QString path)
|
||||||
{
|
{
|
||||||
auto sepIndex = path.indexOf(Tseparator);
|
auto sepIndex = path.indexOf(Tseparator);
|
||||||
if (sepIndex == -1) {
|
if(sepIndex == -1)
|
||||||
|
{
|
||||||
children[path] = SeparatorPrefixTree(true);
|
children[path] = SeparatorPrefixTree(true);
|
||||||
return children[path];
|
return children[path];
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
auto prefix = path.left(sepIndex);
|
auto prefix = path.left(sepIndex);
|
||||||
if (!children.contains(prefix)) {
|
if(!children.contains(prefix))
|
||||||
|
{
|
||||||
children[prefix] = SeparatorPrefixTree(false);
|
children[prefix] = SeparatorPrefixTree(false);
|
||||||
}
|
}
|
||||||
return children[prefix].insert(path.mid(sepIndex + 1));
|
return children[prefix].insert(path.mid(sepIndex + 1));
|
||||||
@ -44,20 +56,26 @@ class SeparatorPrefixTree {
|
|||||||
bool covers(QString path) const
|
bool covers(QString path) const
|
||||||
{
|
{
|
||||||
// if we found some valid node, it's good enough. the tree covers the path
|
// if we found some valid node, it's good enough. the tree covers the path
|
||||||
if (m_contained) {
|
if(m_contained)
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
auto sepIndex = path.indexOf(Tseparator);
|
auto sepIndex = path.indexOf(Tseparator);
|
||||||
if (sepIndex == -1) {
|
if(sepIndex == -1)
|
||||||
|
{
|
||||||
auto found = children.find(path);
|
auto found = children.find(path);
|
||||||
if (found == children.end()) {
|
if(found == children.end())
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return (*found).covers(QString());
|
return (*found).covers(QString());
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
auto prefix = path.left(sepIndex);
|
auto prefix = path.left(sepIndex);
|
||||||
auto found = children.find(prefix);
|
auto found = children.find(prefix);
|
||||||
if (found == children.end()) {
|
if(found == children.end())
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return (*found).covers(path.mid(sepIndex + 1));
|
return (*found).covers(path.mid(sepIndex + 1));
|
||||||
@ -68,33 +86,41 @@ class SeparatorPrefixTree {
|
|||||||
QString cover(QString path) const
|
QString cover(QString path) const
|
||||||
{
|
{
|
||||||
// if we found some valid node, it's good enough. the tree covers the path
|
// if we found some valid node, it's good enough. the tree covers the path
|
||||||
if (m_contained) {
|
if(m_contained)
|
||||||
|
{
|
||||||
return QString("");
|
return QString("");
|
||||||
}
|
}
|
||||||
auto sepIndex = path.indexOf(Tseparator);
|
auto sepIndex = path.indexOf(Tseparator);
|
||||||
if (sepIndex == -1) {
|
if(sepIndex == -1)
|
||||||
|
{
|
||||||
auto found = children.find(path);
|
auto found = children.find(path);
|
||||||
if (found == children.end()) {
|
if(found == children.end())
|
||||||
|
{
|
||||||
return QString();
|
return QString();
|
||||||
}
|
}
|
||||||
auto nested = (*found).cover(QString());
|
auto nested = (*found).cover(QString());
|
||||||
if (nested.isNull()) {
|
if(nested.isNull())
|
||||||
|
{
|
||||||
return nested;
|
return nested;
|
||||||
}
|
}
|
||||||
if (nested.isEmpty())
|
if(nested.isEmpty())
|
||||||
return path;
|
return path;
|
||||||
return path + Tseparator + nested;
|
return path + Tseparator + nested;
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
auto prefix = path.left(sepIndex);
|
auto prefix = path.left(sepIndex);
|
||||||
auto found = children.find(prefix);
|
auto found = children.find(prefix);
|
||||||
if (found == children.end()) {
|
if(found == children.end())
|
||||||
|
{
|
||||||
return QString();
|
return QString();
|
||||||
}
|
}
|
||||||
auto nested = (*found).cover(path.mid(sepIndex + 1));
|
auto nested = (*found).cover(path.mid(sepIndex + 1));
|
||||||
if (nested.isNull()) {
|
if(nested.isNull())
|
||||||
|
{
|
||||||
return nested;
|
return nested;
|
||||||
}
|
}
|
||||||
if (nested.isEmpty())
|
if(nested.isEmpty())
|
||||||
return prefix;
|
return prefix;
|
||||||
return prefix + Tseparator + nested;
|
return prefix + Tseparator + nested;
|
||||||
}
|
}
|
||||||
@ -104,16 +130,21 @@ class SeparatorPrefixTree {
|
|||||||
bool exists(QString path) const
|
bool exists(QString path) const
|
||||||
{
|
{
|
||||||
auto sepIndex = path.indexOf(Tseparator);
|
auto sepIndex = path.indexOf(Tseparator);
|
||||||
if (sepIndex == -1) {
|
if(sepIndex == -1)
|
||||||
|
{
|
||||||
auto found = children.find(path);
|
auto found = children.find(path);
|
||||||
if (found == children.end()) {
|
if(found == children.end())
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
auto prefix = path.left(sepIndex);
|
auto prefix = path.left(sepIndex);
|
||||||
auto found = children.find(prefix);
|
auto found = children.find(prefix);
|
||||||
if (found == children.end()) {
|
if(found == children.end())
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return (*found).exists(path.mid(sepIndex + 1));
|
return (*found).exists(path.mid(sepIndex + 1));
|
||||||
@ -121,19 +152,24 @@ class SeparatorPrefixTree {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// find a node in the tree by name
|
/// find a node in the tree by name
|
||||||
const SeparatorPrefixTree* find(QString path) const
|
const SeparatorPrefixTree * find(QString path) const
|
||||||
{
|
{
|
||||||
auto sepIndex = path.indexOf(Tseparator);
|
auto sepIndex = path.indexOf(Tseparator);
|
||||||
if (sepIndex == -1) {
|
if(sepIndex == -1)
|
||||||
|
{
|
||||||
auto found = children.find(path);
|
auto found = children.find(path);
|
||||||
if (found == children.end()) {
|
if(found == children.end())
|
||||||
|
{
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return &(*found);
|
return &(*found);
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
auto prefix = path.left(sepIndex);
|
auto prefix = path.left(sepIndex);
|
||||||
auto found = children.find(prefix);
|
auto found = children.find(prefix);
|
||||||
if (found == children.end()) {
|
if(found == children.end())
|
||||||
|
{
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
return (*found).find(path.mid(sepIndex + 1));
|
return (*found).find(path.mid(sepIndex + 1));
|
||||||
@ -141,48 +177,70 @@ class SeparatorPrefixTree {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// is this a leaf node?
|
/// is this a leaf node?
|
||||||
bool leaf() const { return children.isEmpty(); }
|
bool leaf() const
|
||||||
|
{
|
||||||
|
return children.isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
/// is this node actually contained in the tree, or is it purely structural?
|
/// is this node actually contained in the tree, or is it purely structural?
|
||||||
bool contained() const { return m_contained; }
|
bool contained() const
|
||||||
|
{
|
||||||
|
return m_contained;
|
||||||
|
}
|
||||||
|
|
||||||
/// Remove a path from the tree
|
/// Remove a path from the tree
|
||||||
bool remove(QString path) { return removeInternal(path) != Failed; }
|
bool remove(QString path)
|
||||||
|
{
|
||||||
|
return removeInternal(path) != Failed;
|
||||||
|
}
|
||||||
|
|
||||||
/// Clear all children of this node tree node
|
/// Clear all children of this node tree node
|
||||||
void clear() { children.clear(); }
|
void clear()
|
||||||
|
{
|
||||||
|
children.clear();
|
||||||
|
}
|
||||||
|
|
||||||
QStringList toStringList() const
|
QStringList toStringList() const
|
||||||
{
|
{
|
||||||
QStringList collected;
|
QStringList collected;
|
||||||
// collecting these is more expensive.
|
// collecting these is more expensive.
|
||||||
auto iter = children.begin();
|
auto iter = children.begin();
|
||||||
while (iter != children.end()) {
|
while(iter != children.end())
|
||||||
|
{
|
||||||
QStringList list = iter.value().toStringList();
|
QStringList list = iter.value().toStringList();
|
||||||
for (int i = 0; i < list.size(); i++) {
|
for(int i = 0; i < list.size(); i++)
|
||||||
|
{
|
||||||
list[i] = iter.key() + Tseparator + list[i];
|
list[i] = iter.key() + Tseparator + list[i];
|
||||||
}
|
}
|
||||||
collected.append(list);
|
collected.append(list);
|
||||||
if ((*iter).m_contained) {
|
if((*iter).m_contained)
|
||||||
|
{
|
||||||
collected.append(iter.key());
|
collected.append(iter.key());
|
||||||
}
|
}
|
||||||
iter++;
|
iter++;
|
||||||
}
|
}
|
||||||
return collected;
|
return collected;
|
||||||
}
|
}
|
||||||
|
private:
|
||||||
private:
|
enum Removal
|
||||||
enum Removal { Failed, Succeeded, HasChildren };
|
{
|
||||||
|
Failed,
|
||||||
|
Succeeded,
|
||||||
|
HasChildren
|
||||||
|
};
|
||||||
Removal removeInternal(QString path = QString())
|
Removal removeInternal(QString path = QString())
|
||||||
{
|
{
|
||||||
if (path.isEmpty()) {
|
if(path.isEmpty())
|
||||||
if (!m_contained) {
|
{
|
||||||
|
if(!m_contained)
|
||||||
|
{
|
||||||
// remove all children - we are removing a prefix
|
// remove all children - we are removing a prefix
|
||||||
clear();
|
clear();
|
||||||
return Succeeded;
|
return Succeeded;
|
||||||
}
|
}
|
||||||
m_contained = false;
|
m_contained = false;
|
||||||
if (children.size()) {
|
if(children.size())
|
||||||
|
{
|
||||||
return HasChildren;
|
return HasChildren;
|
||||||
}
|
}
|
||||||
return Succeeded;
|
return Succeeded;
|
||||||
@ -190,32 +248,42 @@ class SeparatorPrefixTree {
|
|||||||
Removal remStatus = Failed;
|
Removal remStatus = Failed;
|
||||||
QString childToRemove;
|
QString childToRemove;
|
||||||
auto sepIndex = path.indexOf(Tseparator);
|
auto sepIndex = path.indexOf(Tseparator);
|
||||||
if (sepIndex == -1) {
|
if(sepIndex == -1)
|
||||||
|
{
|
||||||
childToRemove = path;
|
childToRemove = path;
|
||||||
auto found = children.find(childToRemove);
|
auto found = children.find(childToRemove);
|
||||||
if (found == children.end()) {
|
if(found == children.end())
|
||||||
|
{
|
||||||
return Failed;
|
return Failed;
|
||||||
}
|
}
|
||||||
remStatus = (*found).removeInternal();
|
remStatus = (*found).removeInternal();
|
||||||
} else {
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
childToRemove = path.left(sepIndex);
|
childToRemove = path.left(sepIndex);
|
||||||
auto found = children.find(childToRemove);
|
auto found = children.find(childToRemove);
|
||||||
if (found == children.end()) {
|
if(found == children.end())
|
||||||
|
{
|
||||||
return Failed;
|
return Failed;
|
||||||
}
|
}
|
||||||
remStatus = (*found).removeInternal(path.mid(sepIndex + 1));
|
remStatus = (*found).removeInternal(path.mid(sepIndex + 1));
|
||||||
}
|
}
|
||||||
switch (remStatus) {
|
switch (remStatus)
|
||||||
|
{
|
||||||
case Failed:
|
case Failed:
|
||||||
case HasChildren: {
|
case HasChildren:
|
||||||
|
{
|
||||||
return remStatus;
|
return remStatus;
|
||||||
}
|
}
|
||||||
case Succeeded: {
|
case Succeeded:
|
||||||
|
{
|
||||||
children.remove(childToRemove);
|
children.remove(childToRemove);
|
||||||
if (m_contained) {
|
if(m_contained)
|
||||||
|
{
|
||||||
return HasChildren;
|
return HasChildren;
|
||||||
}
|
}
|
||||||
if (children.size()) {
|
if(children.size())
|
||||||
|
{
|
||||||
return HasChildren;
|
return HasChildren;
|
||||||
}
|
}
|
||||||
return Succeeded;
|
return Succeeded;
|
||||||
@ -224,7 +292,7 @@ class SeparatorPrefixTree {
|
|||||||
return Failed;
|
return Failed;
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QMap<QString, SeparatorPrefixTree<Tseparator>> children;
|
QMap<QString,SeparatorPrefixTree<Tseparator>> children;
|
||||||
bool m_contained = false;
|
bool m_contained = false;
|
||||||
};
|
};
|
||||||
|
@ -14,16 +14,17 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "SkinUtils.h"
|
#include "SkinUtils.h"
|
||||||
#include "Application.h"
|
|
||||||
#include "net/HttpMetaCache.h"
|
#include "net/HttpMetaCache.h"
|
||||||
|
#include "Application.h"
|
||||||
|
|
||||||
#include <QFile>
|
#include <QFile>
|
||||||
#include <QJsonArray>
|
#include <QPainter>
|
||||||
#include <QJsonDocument>
|
#include <QJsonDocument>
|
||||||
#include <QJsonObject>
|
#include <QJsonObject>
|
||||||
#include <QPainter>
|
#include <QJsonArray>
|
||||||
|
|
||||||
namespace SkinUtils {
|
namespace SkinUtils
|
||||||
|
{
|
||||||
/*
|
/*
|
||||||
* Given a username, return a pixmap of the cached skin (if it exists), QPixmap() otherwise
|
* Given a username, return a pixmap of the cached skin (if it exists), QPixmap() otherwise
|
||||||
*/
|
*/
|
||||||
@ -31,15 +32,12 @@ QPixmap getFaceFromCache(QString username, int height, int width)
|
|||||||
{
|
{
|
||||||
QFile fskin(APPLICATION->metacache()->resolveEntry("skins", username + ".png")->getFullPath());
|
QFile fskin(APPLICATION->metacache()->resolveEntry("skins", username + ".png")->getFullPath());
|
||||||
|
|
||||||
if (fskin.exists()) {
|
if (fskin.exists())
|
||||||
|
{
|
||||||
QPixmap skinTexture(fskin.fileName());
|
QPixmap skinTexture(fskin.fileName());
|
||||||
if (!skinTexture.isNull()) {
|
if(!skinTexture.isNull())
|
||||||
|
{
|
||||||
QPixmap skin = QPixmap(8, 8);
|
QPixmap skin = QPixmap(8, 8);
|
||||||
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
|
||||||
skin.fill(QColorConstants::Transparent);
|
|
||||||
#else
|
|
||||||
skin.fill(QColor(0, 0, 0, 0));
|
|
||||||
#endif
|
|
||||||
QPainter painter(&skin);
|
QPainter painter(&skin);
|
||||||
painter.drawPixmap(0, 0, skinTexture.copy(8, 8, 8, 8));
|
painter.drawPixmap(0, 0, skinTexture.copy(8, 8, 8, 8));
|
||||||
painter.drawPixmap(0, 0, skinTexture.copy(40, 8, 8, 8));
|
painter.drawPixmap(0, 0, skinTexture.copy(40, 8, 8, 8));
|
||||||
@ -49,4 +47,4 @@ QPixmap getFaceFromCache(QString username, int height, int width)
|
|||||||
|
|
||||||
return QPixmap();
|
return QPixmap();
|
||||||
}
|
}
|
||||||
} // namespace SkinUtils
|
}
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
|
|
||||||
#include <QPixmap>
|
#include <QPixmap>
|
||||||
|
|
||||||
namespace SkinUtils {
|
namespace SkinUtils
|
||||||
|
{
|
||||||
QPixmap getFaceFromCache(QString id, int height = 64, int width = 64);
|
QPixmap getFaceFromCache(QString id, int height = 64, int width = 64);
|
||||||
}
|
}
|
||||||
|
@ -35,7 +35,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "StringUtils.h"
|
#include "StringUtils.h"
|
||||||
#include <qpair.h>
|
|
||||||
|
|
||||||
#include <QRegularExpression>
|
#include <QRegularExpression>
|
||||||
#include <QUuid>
|
#include <QUuid>
|
||||||
@ -150,7 +149,7 @@ QString StringUtils::truncateUrlHumanFriendly(QUrl& url, int max_len, bool hard_
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ((url_compact.length() >= max_len) && hard_limit) {
|
if ((url_compact.length() >= max_len) && hard_limit) {
|
||||||
// still too long, truncate normally
|
// still too long, truncate normaly
|
||||||
url_compact = QString(str_url);
|
url_compact = QString(str_url);
|
||||||
auto to_remove = url_compact.length() - max_len + 3;
|
auto to_remove = url_compact.length() - max_len + 3;
|
||||||
url_compact.remove(url_compact.length() - to_remove - 1, to_remove);
|
url_compact.remove(url_compact.length() - to_remove - 1, to_remove);
|
||||||
@ -183,32 +182,3 @@ QString StringUtils::getRandomAlphaNumeric()
|
|||||||
{
|
{
|
||||||
return QUuid::createUuid().toString(QUuid::Id128);
|
return QUuid::createUuid().toString(QUuid::Id128);
|
||||||
}
|
}
|
||||||
|
|
||||||
QPair<QString, QString> 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<QString, QString> 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<QString, QString> 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);
|
|
||||||
}
|
|
||||||
|
@ -36,10 +36,8 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QPair>
|
|
||||||
#include <QString>
|
#include <QString>
|
||||||
#include <QUrl>
|
#include <QUrl>
|
||||||
#include <utility>
|
|
||||||
|
|
||||||
namespace StringUtils {
|
namespace StringUtils {
|
||||||
|
|
||||||
@ -70,19 +68,15 @@ inline QString fromStdString(string s)
|
|||||||
int naturalCompare(const QString& s1, const QString& s2, Qt::CaseSensitivity cs);
|
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
|
* @brief Truncate a url while keeping its readability py placing the `...` in the middle of the path
|
||||||
* @param url Url to truncate
|
* @param url Url to truncate
|
||||||
* @param max_len max length of url in characters
|
* @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 normally.
|
* @param hard_limit if truncating the path can't get the url short enough, truncate it normaly.
|
||||||
*/
|
*/
|
||||||
QString truncateUrlHumanFriendly(QUrl& url, int max_len, bool hard_limit = false);
|
QString truncateUrlHumanFriendly(QUrl &url, int max_len, bool hard_limit = false);
|
||||||
|
|
||||||
QString humanReadableFileSize(double bytes, bool use_si = false, int decimal_points = 1);
|
QString humanReadableFileSize(double bytes, bool use_si = false, int decimal_points = 1);
|
||||||
|
|
||||||
|
|
||||||
QString getRandomAlphaNumeric();
|
QString getRandomAlphaNumeric();
|
||||||
|
|
||||||
QPair<QString, QString> splitFirst(const QString& s, const QString& sep, Qt::CaseSensitivity cs = Qt::CaseSensitive);
|
|
||||||
QPair<QString, QString> splitFirst(const QString& s, QChar sep, Qt::CaseSensitivity cs = Qt::CaseSensitive);
|
|
||||||
QPair<QString, QString> splitFirst(const QString& s, const QRegularExpression& re);
|
|
||||||
|
|
||||||
} // namespace StringUtils
|
} // namespace StringUtils
|
||||||
|
@ -12,20 +12,28 @@ class Usable;
|
|||||||
*
|
*
|
||||||
* @see UseLock
|
* @see UseLock
|
||||||
*/
|
*/
|
||||||
class Usable {
|
class Usable
|
||||||
|
{
|
||||||
friend class UseLock;
|
friend class UseLock;
|
||||||
|
public:
|
||||||
public:
|
std::size_t useCount() const
|
||||||
virtual ~Usable() {}
|
{
|
||||||
|
return m_useCount;
|
||||||
std::size_t useCount() const { return m_useCount; }
|
}
|
||||||
bool isInUse() const { return m_useCount > 0; }
|
bool isInUse() const
|
||||||
|
{
|
||||||
protected:
|
return m_useCount > 0;
|
||||||
virtual void decrementUses() { m_useCount--; }
|
}
|
||||||
virtual void incrementUses() { m_useCount++; }
|
protected:
|
||||||
|
virtual void decrementUses()
|
||||||
private:
|
{
|
||||||
|
m_useCount--;
|
||||||
|
}
|
||||||
|
virtual void incrementUses()
|
||||||
|
{
|
||||||
|
m_useCount++;
|
||||||
|
}
|
||||||
|
private:
|
||||||
std::size_t m_useCount = 0;
|
std::size_t m_useCount = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -34,15 +42,19 @@ class Usable {
|
|||||||
*
|
*
|
||||||
* @see Usable
|
* @see Usable
|
||||||
*/
|
*/
|
||||||
class UseLock {
|
class UseLock
|
||||||
public:
|
{
|
||||||
UseLock(shared_qobject_ptr<Usable> usable) : m_usable(usable)
|
public:
|
||||||
|
UseLock(shared_qobject_ptr<Usable> usable)
|
||||||
|
: m_usable(usable)
|
||||||
{
|
{
|
||||||
// this doesn't use shared pointer use count, because that wouldn't be correct. this count is separate.
|
// this doesn't use shared pointer use count, because that wouldn't be correct. this count is separate.
|
||||||
m_usable->incrementUses();
|
m_usable->incrementUses();
|
||||||
}
|
}
|
||||||
~UseLock() { m_usable->decrementUses(); }
|
~UseLock()
|
||||||
|
{
|
||||||
private:
|
m_usable->decrementUses();
|
||||||
|
}
|
||||||
|
private:
|
||||||
shared_qobject_ptr<Usable> m_usable;
|
shared_qobject_ptr<Usable> m_usable;
|
||||||
};
|
};
|
||||||
|
@ -117,14 +117,12 @@ QDebug operator<<(QDebug debug, const Version& v)
|
|||||||
|
|
||||||
bool first = true;
|
bool first = true;
|
||||||
for (auto s : v.m_sections) {
|
for (auto s : v.m_sections) {
|
||||||
if (!first)
|
if (!first) debug.nospace() << ", ";
|
||||||
debug.nospace() << ", ";
|
|
||||||
debug.nospace() << s.m_fullString;
|
debug.nospace() << s.m_fullString;
|
||||||
first = false;
|
first = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
debug.nospace() << " ]"
|
debug.nospace() << " ]" << " }";
|
||||||
<< " }";
|
|
||||||
|
|
||||||
return debug;
|
return debug;
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2023 flowln <flowlnlnln@gmail.com>
|
* Copyright (C) 2023 flowln <flowlnlnln@gmail.com>
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
*
|
*
|
||||||
@ -48,15 +48,14 @@ class Version {
|
|||||||
Version(QString str);
|
Version(QString str);
|
||||||
Version() = default;
|
Version() = default;
|
||||||
|
|
||||||
bool operator<(const Version& other) const;
|
bool operator<(const Version &other) const;
|
||||||
bool operator<=(const Version& other) const;
|
bool operator<=(const Version &other) const;
|
||||||
bool operator>(const Version& other) const;
|
bool operator>(const Version &other) const;
|
||||||
bool operator>=(const Version& other) const;
|
bool operator>=(const Version &other) const;
|
||||||
bool operator==(const Version& other) const;
|
bool operator==(const Version &other) const;
|
||||||
bool operator!=(const Version& other) const;
|
bool operator!=(const Version &other) const;
|
||||||
|
|
||||||
QString toString() const { return m_string; }
|
QString toString() const { return m_string; }
|
||||||
bool isEmpty() const { return m_string.isEmpty(); }
|
|
||||||
|
|
||||||
friend QDebug operator<<(QDebug debug, const Version& v);
|
friend QDebug operator<<(QDebug debug, const Version& v);
|
||||||
|
|
||||||
@ -64,7 +63,7 @@ class Version {
|
|||||||
struct Section {
|
struct Section {
|
||||||
explicit Section(QString fullString) : m_fullString(std::move(fullString))
|
explicit Section(QString fullString) : m_fullString(std::move(fullString))
|
||||||
{
|
{
|
||||||
qsizetype cutoff = m_fullString.size();
|
int cutoff = m_fullString.size();
|
||||||
for (int i = 0; i < m_fullString.size(); i++) {
|
for (int i = 0; i < m_fullString.size(); i++) {
|
||||||
if (!m_fullString[i].isDigit()) {
|
if (!m_fullString[i].isDigit()) {
|
||||||
cutoff = i;
|
cutoff = i;
|
||||||
@ -73,7 +72,7 @@ class Version {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
||||||
auto numPart = QStringView{ m_fullString }.left(cutoff);
|
auto numPart = QStringView{m_fullString}.left(cutoff);
|
||||||
#else
|
#else
|
||||||
auto numPart = m_fullString.leftRef(cutoff);
|
auto numPart = m_fullString.leftRef(cutoff);
|
||||||
#endif
|
#endif
|
||||||
@ -84,7 +83,7 @@ class Version {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
||||||
auto stringPart = QStringView{ m_fullString }.mid(cutoff);
|
auto stringPart = QStringView{m_fullString}.mid(cutoff);
|
||||||
#else
|
#else
|
||||||
auto stringPart = m_fullString.midRef(cutoff);
|
auto stringPart = m_fullString.midRef(cutoff);
|
||||||
#endif
|
#endif
|
||||||
@ -122,7 +121,7 @@ class Version {
|
|||||||
}
|
}
|
||||||
|
|
||||||
inline bool operator<(const Section& other) const
|
inline bool operator<(const Section& other) const
|
||||||
{
|
{
|
||||||
static auto unequal_is_less = [](Section const& non_null) -> bool {
|
static auto unequal_is_less = [](Section const& non_null) -> bool {
|
||||||
if (non_null.m_stringPart.isEmpty())
|
if (non_null.m_stringPart.isEmpty())
|
||||||
return non_null.m_numPart == 0;
|
return non_null.m_numPart == 0;
|
||||||
@ -151,8 +150,14 @@ class Version {
|
|||||||
return m_fullString < other.m_fullString;
|
return m_fullString < other.m_fullString;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline bool operator!=(const Section& other) const { return !(*this == other); }
|
inline bool operator!=(const Section& other) const
|
||||||
inline bool operator>(const Section& other) const { return !(*this < other || *this == other); }
|
{
|
||||||
|
return !(*this == other);
|
||||||
|
}
|
||||||
|
inline bool operator>(const Section &other) const
|
||||||
|
{
|
||||||
|
return !(*this < other || *this == other);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
@ -161,3 +166,5 @@ class Version {
|
|||||||
|
|
||||||
void parse();
|
void parse();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,8 +1,7 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* Prism Launcher - Minecraft Launcher
|
* PolyMC - Minecraft Launcher
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
* Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me>
|
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -35,48 +34,48 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "VersionProxyModel.h"
|
#include "VersionProxyModel.h"
|
||||||
|
#include "Application.h"
|
||||||
|
#include <QSortFilterProxyModel>
|
||||||
|
#include <QPixmapCache>
|
||||||
#include <Version.h>
|
#include <Version.h>
|
||||||
#include <meta/VersionList.h>
|
#include <meta/VersionList.h>
|
||||||
#include <QPixmapCache>
|
|
||||||
#include <QSortFilterProxyModel>
|
|
||||||
#include "Application.h"
|
|
||||||
|
|
||||||
class VersionFilterModel : public QSortFilterProxyModel {
|
class VersionFilterModel : public QSortFilterProxyModel
|
||||||
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
VersionFilterModel(VersionProxyModel* parent) : QSortFilterProxyModel(parent)
|
VersionFilterModel(VersionProxyModel *parent) : QSortFilterProxyModel(parent)
|
||||||
{
|
{
|
||||||
m_parent = parent;
|
m_parent = parent;
|
||||||
setSortRole(BaseVersionList::SortRole);
|
setSortRole(BaseVersionList::SortRole);
|
||||||
sort(0, Qt::DescendingOrder);
|
sort(0, Qt::DescendingOrder);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool filterAcceptsRow(int source_row, const QModelIndex& source_parent) const
|
bool filterAcceptsRow(int source_row, const QModelIndex &source_parent) const
|
||||||
{
|
{
|
||||||
const auto& filters = m_parent->filters();
|
const auto &filters = m_parent->filters();
|
||||||
const QString& search = m_parent->search();
|
for (auto it = filters.begin(); it != filters.end(); ++it)
|
||||||
const QModelIndex idx = sourceModel()->index(source_row, 0, source_parent);
|
{
|
||||||
|
auto idx = sourceModel()->index(source_row, 0, source_parent);
|
||||||
if (!search.isEmpty() && !sourceModel()->data(idx, BaseVersionList::VersionRole).toString().contains(search, Qt::CaseInsensitive))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
for (auto it = filters.begin(); it != filters.end(); ++it) {
|
|
||||||
auto data = sourceModel()->data(idx, it.key());
|
auto data = sourceModel()->data(idx, it.key());
|
||||||
auto match = data.toString();
|
auto match = data.toString();
|
||||||
if (!it.value()->accepts(match)) {
|
if(!it.value()->accepts(match))
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void filterChanged() { invalidateFilter(); }
|
void filterChanged()
|
||||||
|
{
|
||||||
private:
|
invalidateFilter();
|
||||||
VersionProxyModel* m_parent;
|
}
|
||||||
|
private:
|
||||||
|
VersionProxyModel *m_parent;
|
||||||
};
|
};
|
||||||
|
|
||||||
VersionProxyModel::VersionProxyModel(QObject* parent) : QAbstractProxyModel(parent)
|
VersionProxyModel::VersionProxyModel(QObject *parent) : QAbstractProxyModel(parent)
|
||||||
{
|
{
|
||||||
filterModel = new VersionFilterModel(this);
|
filterModel = new VersionFilterModel(this);
|
||||||
connect(filterModel, &QAbstractItemModel::dataChanged, this, &VersionProxyModel::sourceDataChanged);
|
connect(filterModel, &QAbstractItemModel::dataChanged, this, &VersionProxyModel::sourceDataChanged);
|
||||||
@ -99,17 +98,19 @@ VersionProxyModel::VersionProxyModel(QObject* parent) : QAbstractProxyModel(pare
|
|||||||
|
|
||||||
QVariant VersionProxyModel::headerData(int section, Qt::Orientation orientation, int role) const
|
QVariant VersionProxyModel::headerData(int section, Qt::Orientation orientation, int role) const
|
||||||
{
|
{
|
||||||
if (section < 0 || section >= m_columns.size())
|
if(section < 0 || section >= m_columns.size())
|
||||||
return QVariant();
|
return QVariant();
|
||||||
if (orientation != Qt::Horizontal)
|
if(orientation != Qt::Horizontal)
|
||||||
return QVariant();
|
return QVariant();
|
||||||
auto column = m_columns[section];
|
auto column = m_columns[section];
|
||||||
if (role == Qt::DisplayRole) {
|
if(role == Qt::DisplayRole)
|
||||||
switch (column) {
|
{
|
||||||
|
switch(column)
|
||||||
|
{
|
||||||
case Name:
|
case Name:
|
||||||
return tr("Version");
|
return tr("Version");
|
||||||
case ParentVersion:
|
case ParentVersion:
|
||||||
return tr("Minecraft"); // FIXME: this should come from metadata
|
return tr("Minecraft"); //FIXME: this should come from metadata
|
||||||
case Branch:
|
case Branch:
|
||||||
return tr("Branch");
|
return tr("Branch");
|
||||||
case Type:
|
case Type:
|
||||||
@ -121,12 +122,15 @@ QVariant VersionProxyModel::headerData(int section, Qt::Orientation orientation,
|
|||||||
case Time:
|
case Time:
|
||||||
return tr("Released");
|
return tr("Released");
|
||||||
}
|
}
|
||||||
} else if (role == Qt::ToolTipRole) {
|
}
|
||||||
switch (column) {
|
else if(role == Qt::ToolTipRole)
|
||||||
|
{
|
||||||
|
switch(column)
|
||||||
|
{
|
||||||
case Name:
|
case Name:
|
||||||
return tr("The name of the version.");
|
return tr("The name of the version.");
|
||||||
case ParentVersion:
|
case ParentVersion:
|
||||||
return tr("Minecraft version"); // FIXME: this should come from metadata
|
return tr("Minecraft version"); //FIXME: this should come from metadata
|
||||||
case Branch:
|
case Branch:
|
||||||
return tr("The version's branch");
|
return tr("The version's branch");
|
||||||
case Type:
|
case Type:
|
||||||
@ -142,19 +146,25 @@ QVariant VersionProxyModel::headerData(int section, Qt::Orientation orientation,
|
|||||||
return QVariant();
|
return QVariant();
|
||||||
}
|
}
|
||||||
|
|
||||||
QVariant VersionProxyModel::data(const QModelIndex& index, int role) const
|
QVariant VersionProxyModel::data(const QModelIndex &index, int role) const
|
||||||
{
|
{
|
||||||
if (!index.isValid()) {
|
if(!index.isValid())
|
||||||
|
{
|
||||||
return QVariant();
|
return QVariant();
|
||||||
}
|
}
|
||||||
auto column = m_columns[index.column()];
|
auto column = m_columns[index.column()];
|
||||||
auto parentIndex = mapToSource(index);
|
auto parentIndex = mapToSource(index);
|
||||||
switch (role) {
|
switch(role)
|
||||||
case Qt::DisplayRole: {
|
{
|
||||||
switch (column) {
|
case Qt::DisplayRole:
|
||||||
case Name: {
|
{
|
||||||
|
switch(column)
|
||||||
|
{
|
||||||
|
case Name:
|
||||||
|
{
|
||||||
QString version = sourceModel()->data(parentIndex, BaseVersionList::VersionRole).toString();
|
QString version = sourceModel()->data(parentIndex, BaseVersionList::VersionRole).toString();
|
||||||
if (version == m_currentVersion) {
|
if(version == m_currentVersion)
|
||||||
|
{
|
||||||
return tr("%1 (installed)").arg(version);
|
return tr("%1 (installed)").arg(version);
|
||||||
}
|
}
|
||||||
return version;
|
return version;
|
||||||
@ -175,38 +185,58 @@ QVariant VersionProxyModel::data(const QModelIndex& index, int role) const
|
|||||||
return QVariant();
|
return QVariant();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
case Qt::ToolTipRole: {
|
case Qt::ToolTipRole:
|
||||||
if (column == Name && hasRecommended) {
|
{
|
||||||
|
if(column == Name && hasRecommended)
|
||||||
|
{
|
||||||
auto value = sourceModel()->data(parentIndex, BaseVersionList::RecommendedRole);
|
auto value = sourceModel()->data(parentIndex, BaseVersionList::RecommendedRole);
|
||||||
if (value.toBool()) {
|
if(value.toBool())
|
||||||
|
{
|
||||||
return tr("Recommended");
|
return tr("Recommended");
|
||||||
} else if (hasLatest) {
|
} else if(hasLatest) {
|
||||||
auto value = sourceModel()->data(parentIndex, BaseVersionList::LatestRole);
|
auto value = sourceModel()->data(parentIndex, BaseVersionList::LatestRole);
|
||||||
if (value.toBool()) {
|
if(value.toBool())
|
||||||
|
{
|
||||||
return tr("Latest");
|
return tr("Latest");
|
||||||
}
|
}
|
||||||
|
} else if(index.row() == 0)
|
||||||
|
{
|
||||||
|
return tr("Latest");
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return sourceModel()->data(parentIndex, BaseVersionList::VersionIdRole);
|
return sourceModel()->data(parentIndex, BaseVersionList::VersionIdRole);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
case Qt::DecorationRole: {
|
case Qt::DecorationRole:
|
||||||
switch (column) {
|
{
|
||||||
case Name: {
|
switch(column)
|
||||||
if (hasRecommended) {
|
{
|
||||||
auto recommenced = sourceModel()->data(parentIndex, BaseVersionList::RecommendedRole);
|
case Name:
|
||||||
if (recommenced.toBool()) {
|
{
|
||||||
|
if(hasRecommended)
|
||||||
|
{
|
||||||
|
auto value = sourceModel()->data(parentIndex, BaseVersionList::RecommendedRole);
|
||||||
|
if(value.toBool())
|
||||||
|
{
|
||||||
return APPLICATION->getThemedIcon("star");
|
return APPLICATION->getThemedIcon("star");
|
||||||
} else if (hasLatest) {
|
}
|
||||||
auto latest = sourceModel()->data(parentIndex, BaseVersionList::LatestRole);
|
else if(hasLatest)
|
||||||
if (latest.toBool()) {
|
{
|
||||||
|
auto value = sourceModel()->data(parentIndex, BaseVersionList::LatestRole);
|
||||||
|
if(value.toBool())
|
||||||
|
{
|
||||||
return APPLICATION->getThemedIcon("bug");
|
return APPLICATION->getThemedIcon("bug");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else if(index.row() == 0)
|
||||||
|
{
|
||||||
|
return APPLICATION->getThemedIcon("bug");
|
||||||
|
}
|
||||||
QPixmap pixmap;
|
QPixmap pixmap;
|
||||||
QPixmapCache::find("placeholder", &pixmap);
|
QPixmapCache::find("placeholder", &pixmap);
|
||||||
if (!pixmap) {
|
if(!pixmap)
|
||||||
QPixmap px(16, 16);
|
{
|
||||||
|
QPixmap px(16,16);
|
||||||
px.fill(Qt::transparent);
|
px.fill(Qt::transparent);
|
||||||
QPixmapCache::insert("placeholder", px);
|
QPixmapCache::insert("placeholder", px);
|
||||||
return px;
|
return px;
|
||||||
@ -214,13 +244,16 @@ QVariant VersionProxyModel::data(const QModelIndex& index, int role) const
|
|||||||
return pixmap;
|
return pixmap;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
default: {
|
default:
|
||||||
|
{
|
||||||
return QVariant();
|
return QVariant();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
default: {
|
default:
|
||||||
if (roles.contains((BaseVersionList::ModelRoles)role)) {
|
{
|
||||||
|
if(roles.contains((BaseVersionList::ModelRoles)role))
|
||||||
|
{
|
||||||
return sourceModel()->data(parentIndex, role);
|
return sourceModel()->data(parentIndex, role);
|
||||||
}
|
}
|
||||||
return QVariant();
|
return QVariant();
|
||||||
@ -228,56 +261,61 @@ QVariant VersionProxyModel::data(const QModelIndex& index, int role) const
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
QModelIndex VersionProxyModel::parent([[maybe_unused]] const QModelIndex& child) const
|
QModelIndex VersionProxyModel::parent(const QModelIndex &child) const
|
||||||
{
|
{
|
||||||
return QModelIndex();
|
return QModelIndex();
|
||||||
}
|
}
|
||||||
|
|
||||||
QModelIndex VersionProxyModel::mapFromSource(const QModelIndex& sourceIndex) const
|
QModelIndex VersionProxyModel::mapFromSource(const QModelIndex &sourceIndex) const
|
||||||
{
|
{
|
||||||
if (sourceIndex.isValid()) {
|
if(sourceIndex.isValid())
|
||||||
|
{
|
||||||
return index(sourceIndex.row(), 0);
|
return index(sourceIndex.row(), 0);
|
||||||
}
|
}
|
||||||
return QModelIndex();
|
return QModelIndex();
|
||||||
}
|
}
|
||||||
|
|
||||||
QModelIndex VersionProxyModel::mapToSource(const QModelIndex& proxyIndex) const
|
QModelIndex VersionProxyModel::mapToSource(const QModelIndex &proxyIndex) const
|
||||||
{
|
{
|
||||||
if (proxyIndex.isValid()) {
|
if(proxyIndex.isValid())
|
||||||
|
{
|
||||||
return sourceModel()->index(proxyIndex.row(), 0);
|
return sourceModel()->index(proxyIndex.row(), 0);
|
||||||
}
|
}
|
||||||
return QModelIndex();
|
return QModelIndex();
|
||||||
}
|
}
|
||||||
|
|
||||||
QModelIndex VersionProxyModel::index(int row, int column, const QModelIndex& parent) const
|
QModelIndex VersionProxyModel::index(int row, int column, const QModelIndex &parent) const
|
||||||
{
|
{
|
||||||
// no trees here... shoo
|
// no trees here... shoo
|
||||||
if (parent.isValid()) {
|
if(parent.isValid())
|
||||||
|
{
|
||||||
return QModelIndex();
|
return QModelIndex();
|
||||||
}
|
}
|
||||||
if (row < 0 || row >= sourceModel()->rowCount())
|
if(row < 0 || row >= sourceModel()->rowCount())
|
||||||
return QModelIndex();
|
return QModelIndex();
|
||||||
if (column < 0 || column >= columnCount())
|
if(column < 0 || column >= columnCount())
|
||||||
return QModelIndex();
|
return QModelIndex();
|
||||||
return QAbstractItemModel::createIndex(row, column);
|
return QAbstractItemModel::createIndex(row, column);
|
||||||
}
|
}
|
||||||
|
|
||||||
int VersionProxyModel::columnCount(const QModelIndex& parent) const
|
int VersionProxyModel::columnCount(const QModelIndex &parent) const
|
||||||
{
|
{
|
||||||
return parent.isValid() ? 0 : m_columns.size();
|
return parent.isValid() ? 0 : m_columns.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
int VersionProxyModel::rowCount(const QModelIndex& parent) const
|
int VersionProxyModel::rowCount(const QModelIndex &parent) const
|
||||||
{
|
{
|
||||||
if (sourceModel()) {
|
if(sourceModel())
|
||||||
|
{
|
||||||
return sourceModel()->rowCount(parent);
|
return sourceModel()->rowCount(parent);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void VersionProxyModel::sourceDataChanged(const QModelIndex& source_top_left, const QModelIndex& source_bottom_right)
|
void VersionProxyModel::sourceDataChanged(const QModelIndex &source_top_left,
|
||||||
|
const QModelIndex &source_bottom_right)
|
||||||
{
|
{
|
||||||
if (source_top_left.parent() != source_bottom_right.parent())
|
if(source_top_left.parent() != source_bottom_right.parent())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// whole row is getting changed
|
// whole row is getting changed
|
||||||
@ -286,20 +324,22 @@ void VersionProxyModel::sourceDataChanged(const QModelIndex& source_top_left, co
|
|||||||
emit dataChanged(topLeft, bottomRight);
|
emit dataChanged(topLeft, bottomRight);
|
||||||
}
|
}
|
||||||
|
|
||||||
void VersionProxyModel::setSourceModel(QAbstractItemModel* replacingRaw)
|
void VersionProxyModel::setSourceModel(QAbstractItemModel *replacingRaw)
|
||||||
{
|
{
|
||||||
auto replacing = dynamic_cast<BaseVersionList*>(replacingRaw);
|
auto replacing = dynamic_cast<BaseVersionList *>(replacingRaw);
|
||||||
beginResetModel();
|
beginResetModel();
|
||||||
|
|
||||||
m_columns.clear();
|
m_columns.clear();
|
||||||
if (!replacing) {
|
if(!replacing)
|
||||||
|
{
|
||||||
roles.clear();
|
roles.clear();
|
||||||
filterModel->setSourceModel(replacing);
|
filterModel->setSourceModel(replacing);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
roles = replacing->providesRoles();
|
roles = replacing->providesRoles();
|
||||||
if (roles.contains(BaseVersionList::VersionRole)) {
|
if(roles.contains(BaseVersionList::VersionRole))
|
||||||
|
{
|
||||||
m_columns.push_back(Name);
|
m_columns.push_back(Name);
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
@ -308,25 +348,32 @@ void VersionProxyModel::setSourceModel(QAbstractItemModel* replacingRaw)
|
|||||||
m_columns.push_back(ParentVersion);
|
m_columns.push_back(ParentVersion);
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
if (roles.contains(BaseVersionList::ArchitectureRole)) {
|
if(roles.contains(BaseVersionList::ArchitectureRole))
|
||||||
|
{
|
||||||
m_columns.push_back(Architecture);
|
m_columns.push_back(Architecture);
|
||||||
}
|
}
|
||||||
if (roles.contains(BaseVersionList::PathRole)) {
|
if(roles.contains(BaseVersionList::PathRole))
|
||||||
|
{
|
||||||
m_columns.push_back(Path);
|
m_columns.push_back(Path);
|
||||||
}
|
}
|
||||||
if (roles.contains(Meta::VersionList::TimeRole)) {
|
if(roles.contains(Meta::VersionList::TimeRole))
|
||||||
|
{
|
||||||
m_columns.push_back(Time);
|
m_columns.push_back(Time);
|
||||||
}
|
}
|
||||||
if (roles.contains(BaseVersionList::BranchRole)) {
|
if(roles.contains(BaseVersionList::BranchRole))
|
||||||
|
{
|
||||||
m_columns.push_back(Branch);
|
m_columns.push_back(Branch);
|
||||||
}
|
}
|
||||||
if (roles.contains(BaseVersionList::TypeRole)) {
|
if(roles.contains(BaseVersionList::TypeRole))
|
||||||
|
{
|
||||||
m_columns.push_back(Type);
|
m_columns.push_back(Type);
|
||||||
}
|
}
|
||||||
if (roles.contains(BaseVersionList::RecommendedRole)) {
|
if(roles.contains(BaseVersionList::RecommendedRole))
|
||||||
|
{
|
||||||
hasRecommended = true;
|
hasRecommended = true;
|
||||||
}
|
}
|
||||||
if (roles.contains(BaseVersionList::LatestRole)) {
|
if(roles.contains(BaseVersionList::LatestRole))
|
||||||
|
{
|
||||||
hasLatest = true;
|
hasLatest = true;
|
||||||
}
|
}
|
||||||
filterModel->setSourceModel(replacing);
|
filterModel->setSourceModel(replacing);
|
||||||
@ -336,13 +383,16 @@ void VersionProxyModel::setSourceModel(QAbstractItemModel* replacingRaw)
|
|||||||
|
|
||||||
QModelIndex VersionProxyModel::getRecommended() const
|
QModelIndex VersionProxyModel::getRecommended() const
|
||||||
{
|
{
|
||||||
if (!roles.contains(BaseVersionList::RecommendedRole)) {
|
if(!roles.contains(BaseVersionList::RecommendedRole))
|
||||||
|
{
|
||||||
return index(0, 0);
|
return index(0, 0);
|
||||||
}
|
}
|
||||||
int recommended = 0;
|
int recommended = 0;
|
||||||
for (int i = 0; i < rowCount(); i++) {
|
for (int i = 0; i < rowCount(); i++)
|
||||||
|
{
|
||||||
auto value = sourceModel()->data(mapToSource(index(i, 0)), BaseVersionList::RecommendedRole);
|
auto value = sourceModel()->data(mapToSource(index(i, 0)), BaseVersionList::RecommendedRole);
|
||||||
if (value.toBool()) {
|
if (value.toBool())
|
||||||
|
{
|
||||||
recommended = i;
|
recommended = i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -352,13 +402,16 @@ QModelIndex VersionProxyModel::getRecommended() const
|
|||||||
QModelIndex VersionProxyModel::getVersion(const QString& version) const
|
QModelIndex VersionProxyModel::getVersion(const QString& version) const
|
||||||
{
|
{
|
||||||
int found = -1;
|
int found = -1;
|
||||||
for (int i = 0; i < rowCount(); i++) {
|
for (int i = 0; i < rowCount(); i++)
|
||||||
|
{
|
||||||
auto value = sourceModel()->data(mapToSource(index(i, 0)), BaseVersionList::VersionRole);
|
auto value = sourceModel()->data(mapToSource(index(i, 0)), BaseVersionList::VersionRole);
|
||||||
if (value.toString() == version) {
|
if (value.toString() == version)
|
||||||
|
{
|
||||||
found = i;
|
found = i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (found == -1) {
|
if(found == -1)
|
||||||
|
{
|
||||||
return QModelIndex();
|
return QModelIndex();
|
||||||
}
|
}
|
||||||
return index(found, 0);
|
return index(found, 0);
|
||||||
@ -367,32 +420,20 @@ QModelIndex VersionProxyModel::getVersion(const QString& version) const
|
|||||||
void VersionProxyModel::clearFilters()
|
void VersionProxyModel::clearFilters()
|
||||||
{
|
{
|
||||||
m_filters.clear();
|
m_filters.clear();
|
||||||
m_search.clear();
|
|
||||||
filterModel->filterChanged();
|
filterModel->filterChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
void VersionProxyModel::setFilter(const BaseVersionList::ModelRoles column, Filter* f)
|
void VersionProxyModel::setFilter(const BaseVersionList::ModelRoles column, Filter * f)
|
||||||
{
|
{
|
||||||
m_filters[column].reset(f);
|
m_filters[column].reset(f);
|
||||||
filterModel->filterChanged();
|
filterModel->filterChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
void VersionProxyModel::setSearch(const QString& search)
|
const VersionProxyModel::FilterMap &VersionProxyModel::filters() const
|
||||||
{
|
|
||||||
m_search = search;
|
|
||||||
filterModel->filterChanged();
|
|
||||||
}
|
|
||||||
|
|
||||||
const VersionProxyModel::FilterMap& VersionProxyModel::filters() const
|
|
||||||
{
|
{
|
||||||
return m_filters;
|
return m_filters;
|
||||||
}
|
}
|
||||||
|
|
||||||
const QString& VersionProxyModel::search() const
|
|
||||||
{
|
|
||||||
return m_search;
|
|
||||||
}
|
|
||||||
|
|
||||||
void VersionProxyModel::sourceAboutToBeReset()
|
void VersionProxyModel::sourceAboutToBeReset()
|
||||||
{
|
{
|
||||||
beginResetModel();
|
beginResetModel();
|
||||||
@ -408,9 +449,7 @@ void VersionProxyModel::sourceRowsAboutToBeInserted(const QModelIndex& parent, i
|
|||||||
beginInsertRows(parent, first, last);
|
beginInsertRows(parent, first, last);
|
||||||
}
|
}
|
||||||
|
|
||||||
void VersionProxyModel::sourceRowsInserted([[maybe_unused]] const QModelIndex& parent,
|
void VersionProxyModel::sourceRowsInserted(const QModelIndex& parent, int first, int last)
|
||||||
[[maybe_unused]] int first,
|
|
||||||
[[maybe_unused]] int last)
|
|
||||||
{
|
{
|
||||||
endInsertRows();
|
endInsertRows();
|
||||||
}
|
}
|
||||||
@ -420,12 +459,12 @@ void VersionProxyModel::sourceRowsAboutToBeRemoved(const QModelIndex& parent, in
|
|||||||
beginRemoveRows(parent, first, last);
|
beginRemoveRows(parent, first, last);
|
||||||
}
|
}
|
||||||
|
|
||||||
void VersionProxyModel::sourceRowsRemoved([[maybe_unused]] const QModelIndex& parent, [[maybe_unused]] int first, [[maybe_unused]] int last)
|
void VersionProxyModel::sourceRowsRemoved(const QModelIndex& parent, int first, int last)
|
||||||
{
|
{
|
||||||
endRemoveRows();
|
endRemoveRows();
|
||||||
}
|
}
|
||||||
|
|
||||||
void VersionProxyModel::setCurrentVersion(const QString& version)
|
void VersionProxyModel::setCurrentVersion(const QString &version)
|
||||||
{
|
{
|
||||||
m_currentVersion = version;
|
m_currentVersion = version;
|
||||||
}
|
}
|
||||||
|
@ -6,53 +6,61 @@
|
|||||||
|
|
||||||
class VersionFilterModel;
|
class VersionFilterModel;
|
||||||
|
|
||||||
class VersionProxyModel : public QAbstractProxyModel {
|
class VersionProxyModel: public QAbstractProxyModel
|
||||||
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
enum Column { Name, ParentVersion, Branch, Type, Architecture, Path, Time };
|
|
||||||
using FilterMap = QHash<BaseVersionList::ModelRoles, std::shared_ptr<Filter>>;
|
|
||||||
|
|
||||||
public:
|
enum Column
|
||||||
VersionProxyModel(QObject* parent = 0);
|
{
|
||||||
virtual ~VersionProxyModel(){};
|
Name,
|
||||||
|
ParentVersion,
|
||||||
|
Branch,
|
||||||
|
Type,
|
||||||
|
Architecture,
|
||||||
|
Path,
|
||||||
|
Time
|
||||||
|
};
|
||||||
|
typedef QHash<BaseVersionList::ModelRoles, std::shared_ptr<Filter>> FilterMap;
|
||||||
|
|
||||||
virtual int columnCount(const QModelIndex& parent = QModelIndex()) const override;
|
public:
|
||||||
virtual int rowCount(const QModelIndex& parent = QModelIndex()) const override;
|
VersionProxyModel ( QObject* parent = 0 );
|
||||||
virtual QModelIndex mapFromSource(const QModelIndex& sourceIndex) const override;
|
virtual ~VersionProxyModel() {};
|
||||||
virtual QModelIndex mapToSource(const QModelIndex& proxyIndex) const override;
|
|
||||||
virtual QModelIndex index(int row, int column, const QModelIndex& parent = QModelIndex()) const override;
|
virtual int columnCount(const QModelIndex &parent = QModelIndex()) const override;
|
||||||
virtual QVariant data(const QModelIndex& index, int role = Qt::DisplayRole) const override;
|
virtual int rowCount(const QModelIndex &parent = QModelIndex()) const override;
|
||||||
|
virtual QModelIndex mapFromSource(const QModelIndex &sourceIndex) const override;
|
||||||
|
virtual QModelIndex mapToSource(const QModelIndex &proxyIndex) const override;
|
||||||
|
virtual QModelIndex index(int row, int column, const QModelIndex &parent = QModelIndex()) const override;
|
||||||
|
virtual QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
|
||||||
virtual QVariant headerData(int section, Qt::Orientation orientation, int role = Qt::DisplayRole) const override;
|
virtual QVariant headerData(int section, Qt::Orientation orientation, int role = Qt::DisplayRole) const override;
|
||||||
virtual QModelIndex parent(const QModelIndex& child) const override;
|
virtual QModelIndex parent(const QModelIndex &child) const override;
|
||||||
virtual void setSourceModel(QAbstractItemModel* sourceModel) override;
|
virtual void setSourceModel(QAbstractItemModel *sourceModel) override;
|
||||||
|
|
||||||
const FilterMap& filters() const;
|
const FilterMap &filters() const;
|
||||||
const QString& search() const;
|
void setFilter(const BaseVersionList::ModelRoles column, Filter * filter);
|
||||||
void setFilter(const BaseVersionList::ModelRoles column, Filter* filter);
|
|
||||||
void setSearch(const QString& search);
|
|
||||||
void clearFilters();
|
void clearFilters();
|
||||||
QModelIndex getRecommended() const;
|
QModelIndex getRecommended() const;
|
||||||
QModelIndex getVersion(const QString& version) const;
|
QModelIndex getVersion(const QString & version) const;
|
||||||
void setCurrentVersion(const QString& version);
|
void setCurrentVersion(const QString &version);
|
||||||
private slots:
|
private slots:
|
||||||
|
|
||||||
void sourceDataChanged(const QModelIndex& source_top_left, const QModelIndex& source_bottom_right);
|
void sourceDataChanged(const QModelIndex &source_top_left,const QModelIndex &source_bottom_right);
|
||||||
|
|
||||||
void sourceAboutToBeReset();
|
void sourceAboutToBeReset();
|
||||||
void sourceReset();
|
void sourceReset();
|
||||||
|
|
||||||
void sourceRowsAboutToBeInserted(const QModelIndex& parent, int first, int last);
|
void sourceRowsAboutToBeInserted(const QModelIndex &parent, int first, int last);
|
||||||
void sourceRowsInserted(const QModelIndex& parent, int first, int last);
|
void sourceRowsInserted(const QModelIndex &parent, int first, int last);
|
||||||
|
|
||||||
void sourceRowsAboutToBeRemoved(const QModelIndex& parent, int first, int last);
|
void sourceRowsAboutToBeRemoved(const QModelIndex &parent, int first, int last);
|
||||||
void sourceRowsRemoved(const QModelIndex& parent, int first, int last);
|
void sourceRowsRemoved(const QModelIndex &parent, int first, int last);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QList<Column> m_columns;
|
QList<Column> m_columns;
|
||||||
FilterMap m_filters;
|
FilterMap m_filters;
|
||||||
QString m_search;
|
|
||||||
BaseVersionList::RoleList roles;
|
BaseVersionList::RoleList roles;
|
||||||
VersionFilterModel* filterModel;
|
VersionFilterModel * filterModel;
|
||||||
bool hasRecommended = false;
|
bool hasRecommended = false;
|
||||||
bool hasLatest = false;
|
bool hasLatest = false;
|
||||||
QString m_currentVersion;
|
QString m_currentVersion;
|
||||||
|
@ -1,15 +1,20 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QFileSystemWatcher>
|
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
#include <QFileSystemWatcher>
|
||||||
|
|
||||||
struct WatchLock {
|
struct WatchLock
|
||||||
WatchLock(QFileSystemWatcher* watcher, const QString& directory) : m_watcher(watcher), m_directory(directory)
|
{
|
||||||
|
WatchLock(QFileSystemWatcher * watcher, const QString& directory)
|
||||||
|
: m_watcher(watcher), m_directory(directory)
|
||||||
{
|
{
|
||||||
m_watcher->removePath(m_directory);
|
m_watcher->removePath(m_directory);
|
||||||
}
|
}
|
||||||
~WatchLock() { m_watcher->addPath(m_directory); }
|
~WatchLock()
|
||||||
QFileSystemWatcher* m_watcher;
|
{
|
||||||
|
m_watcher->addPath(m_directory);
|
||||||
|
}
|
||||||
|
QFileSystemWatcher * m_watcher;
|
||||||
QString m_directory;
|
QString m_directory;
|
||||||
};
|
};
|
||||||
|
@ -1,128 +0,0 @@
|
|||||||
/*
|
|
||||||
* Prism Launcher - Minecraft Launcher
|
|
||||||
* Copyright (C) 2022 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 <https://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef WIN32_LEAN_AND_MEAN
|
|
||||||
#define WIN32_LEAN_AND_MEAN
|
|
||||||
#endif
|
|
||||||
#include <fcntl.h>
|
|
||||||
#include <io.h>
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <windows.h>
|
|
||||||
#include <iostream>
|
|
||||||
|
|
||||||
void RedirectHandle(DWORD handle, FILE* stream, const char* mode)
|
|
||||||
{
|
|
||||||
HANDLE stdHandle = GetStdHandle(handle);
|
|
||||||
if (stdHandle != INVALID_HANDLE_VALUE) {
|
|
||||||
int fileDescriptor = _open_osfhandle((intptr_t)stdHandle, _O_TEXT);
|
|
||||||
if (fileDescriptor != -1) {
|
|
||||||
FILE* file = _fdopen(fileDescriptor, mode);
|
|
||||||
if (file != NULL) {
|
|
||||||
int dup2Result = _dup2(_fileno(file), _fileno(stream));
|
|
||||||
if (dup2Result == 0) {
|
|
||||||
setvbuf(stream, NULL, _IONBF, 0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// taken from https://stackoverflow.com/a/25927081
|
|
||||||
// getting a proper output to console with redirection support on windows is apparently hell
|
|
||||||
void BindCrtHandlesToStdHandles(bool bindStdIn, bool bindStdOut, bool bindStdErr)
|
|
||||||
{
|
|
||||||
// Re-initialize the C runtime "FILE" handles with clean handles bound to "nul". We do this because it has been
|
|
||||||
// observed that the file number of our standard handle file objects can be assigned internally to a value of -2
|
|
||||||
// when not bound to a valid target, which represents some kind of unknown internal invalid state. In this state our
|
|
||||||
// call to "_dup2" fails, as it specifically tests to ensure that the target file number isn't equal to this value
|
|
||||||
// before allowing the operation to continue. We can resolve this issue by first "re-opening" the target files to
|
|
||||||
// use the "nul" device, which will place them into a valid state, after which we can redirect them to our target
|
|
||||||
// using the "_dup2" function.
|
|
||||||
if (bindStdIn) {
|
|
||||||
FILE* dummyFile;
|
|
||||||
freopen_s(&dummyFile, "nul", "r", stdin);
|
|
||||||
}
|
|
||||||
if (bindStdOut) {
|
|
||||||
FILE* dummyFile;
|
|
||||||
freopen_s(&dummyFile, "nul", "w", stdout);
|
|
||||||
}
|
|
||||||
if (bindStdErr) {
|
|
||||||
FILE* dummyFile;
|
|
||||||
freopen_s(&dummyFile, "nul", "w", stderr);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Redirect unbuffered stdin from the current standard input handle
|
|
||||||
if (bindStdIn) {
|
|
||||||
RedirectHandle(STD_INPUT_HANDLE, stdin, "r");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Redirect unbuffered stdout to the current standard output handle
|
|
||||||
if (bindStdOut) {
|
|
||||||
RedirectHandle(STD_OUTPUT_HANDLE, stdout, "w");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Redirect unbuffered stderr to the current standard error handle
|
|
||||||
if (bindStdErr) {
|
|
||||||
RedirectHandle(STD_ERROR_HANDLE, stderr, "w");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Clear the error state for each of the C++ standard stream objects. We need to do this, as attempts to access the
|
|
||||||
// standard streams before they refer to a valid target will cause the iostream objects to enter an error state. In
|
|
||||||
// versions of Visual Studio after 2005, this seems to always occur during startup regardless of whether anything
|
|
||||||
// has been read from or written to the targets or not.
|
|
||||||
if (bindStdIn) {
|
|
||||||
std::wcin.clear();
|
|
||||||
std::cin.clear();
|
|
||||||
}
|
|
||||||
if (bindStdOut) {
|
|
||||||
std::wcout.clear();
|
|
||||||
std::cout.clear();
|
|
||||||
}
|
|
||||||
if (bindStdErr) {
|
|
||||||
std::wcerr.clear();
|
|
||||||
std::cerr.clear();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool AttachWindowsConsole()
|
|
||||||
{
|
|
||||||
auto stdinType = GetFileType(GetStdHandle(STD_INPUT_HANDLE));
|
|
||||||
auto stdoutType = GetFileType(GetStdHandle(STD_OUTPUT_HANDLE));
|
|
||||||
auto stderrType = GetFileType(GetStdHandle(STD_ERROR_HANDLE));
|
|
||||||
|
|
||||||
bool bindStdIn = false;
|
|
||||||
bool bindStdOut = false;
|
|
||||||
bool bindStdErr = false;
|
|
||||||
|
|
||||||
if (stdinType == FILE_TYPE_CHAR || stdinType == FILE_TYPE_UNKNOWN) {
|
|
||||||
bindStdIn = true;
|
|
||||||
}
|
|
||||||
if (stdoutType == FILE_TYPE_CHAR || stdoutType == FILE_TYPE_UNKNOWN) {
|
|
||||||
bindStdOut = true;
|
|
||||||
}
|
|
||||||
if (stderrType == FILE_TYPE_CHAR || stderrType == FILE_TYPE_UNKNOWN) {
|
|
||||||
bindStdErr = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (AttachConsole(ATTACH_PARENT_PROCESS)) {
|
|
||||||
BindCrtHandlesToStdHandles(bindStdIn, bindStdOut, bindStdErr);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
@ -37,7 +37,11 @@
|
|||||||
#include <sys.h>
|
#include <sys.h>
|
||||||
|
|
||||||
#if defined Q_OS_WIN32
|
#if defined Q_OS_WIN32
|
||||||
#include "WindowsConsole.h"
|
#ifndef WIN32_LEAN_AND_MEAN
|
||||||
|
#define WIN32_LEAN_AND_MEAN
|
||||||
|
#endif
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <windows.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Snippet from https://github.com/gulrak/filesystem#using-it-as-single-file-header
|
// Snippet from https://github.com/gulrak/filesystem#using-it-as-single-file-header
|
||||||
@ -63,7 +67,21 @@ FileLinkApp::FileLinkApp(int& argc, char** argv) : QCoreApplication(argc, argv),
|
|||||||
{
|
{
|
||||||
#if defined Q_OS_WIN32
|
#if defined Q_OS_WIN32
|
||||||
// attach the parent console
|
// attach the parent console
|
||||||
if (AttachWindowsConsole()) {
|
if (AttachConsole(ATTACH_PARENT_PROCESS)) {
|
||||||
|
// if attach succeeds, reopen and sync all the i/o
|
||||||
|
if (freopen("CON", "w", stdout)) {
|
||||||
|
std::cout.sync_with_stdio();
|
||||||
|
}
|
||||||
|
if (freopen("CON", "w", stderr)) {
|
||||||
|
std::cerr.sync_with_stdio();
|
||||||
|
}
|
||||||
|
if (freopen("CON", "r", stdin)) {
|
||||||
|
std::cin.sync_with_stdio();
|
||||||
|
}
|
||||||
|
auto out = GetStdHandle(STD_OUTPUT_HANDLE);
|
||||||
|
DWORD written;
|
||||||
|
const char* endline = "\n";
|
||||||
|
WriteConsole(out, endline, strlen(endline), &written, NULL);
|
||||||
consoleAttached = true;
|
consoleAttached = true;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@ -93,7 +111,6 @@ FileLinkApp::FileLinkApp(int& argc, char** argv) : QCoreApplication(argc, argv),
|
|||||||
joinServer(serverToJoin);
|
joinServer(serverToJoin);
|
||||||
} else {
|
} else {
|
||||||
qDebug() << "no server to join";
|
qDebug() << "no server to join";
|
||||||
m_status = Failed;
|
|
||||||
exit();
|
exit();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -109,7 +126,6 @@ void FileLinkApp::joinServer(QString server)
|
|||||||
connect(&socket, &QLocalSocket::readyRead, this, &FileLinkApp::readPathPairs);
|
connect(&socket, &QLocalSocket::readyRead, this, &FileLinkApp::readPathPairs);
|
||||||
|
|
||||||
connect(&socket, &QLocalSocket::errorOccurred, this, [&](QLocalSocket::LocalSocketError socketError) {
|
connect(&socket, &QLocalSocket::errorOccurred, this, [&](QLocalSocket::LocalSocketError socketError) {
|
||||||
m_status = Failed;
|
|
||||||
switch (socketError) {
|
switch (socketError) {
|
||||||
case QLocalSocket::ServerNotFoundError:
|
case QLocalSocket::ServerNotFoundError:
|
||||||
qDebug()
|
qDebug()
|
||||||
@ -134,7 +150,6 @@ void FileLinkApp::joinServer(QString server)
|
|||||||
|
|
||||||
connect(&socket, &QLocalSocket::disconnected, this, [&]() {
|
connect(&socket, &QLocalSocket::disconnected, this, [&]() {
|
||||||
qDebug() << "disconnected from server, should exit";
|
qDebug() << "disconnected from server, should exit";
|
||||||
m_status = Succeeded;
|
|
||||||
exit();
|
exit();
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -173,7 +188,7 @@ void FileLinkApp::runLink()
|
|||||||
FS::LinkResult result = { src_path, dst_path, QString::fromStdString(os_err.message()), os_err.value() };
|
FS::LinkResult result = { src_path, dst_path, QString::fromStdString(os_err.message()), os_err.value() };
|
||||||
m_path_results.append(result);
|
m_path_results.append(result);
|
||||||
} else {
|
} else {
|
||||||
FS::LinkResult result = { src_path, dst_path, "", 0 };
|
FS::LinkResult result = { src_path, dst_path };
|
||||||
m_path_results.append(result);
|
m_path_results.append(result);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -233,7 +248,7 @@ void FileLinkApp::readPathPairs()
|
|||||||
in >> numLinks;
|
in >> numLinks;
|
||||||
qDebug() << "numLinks" << numLinks;
|
qDebug() << "numLinks" << numLinks;
|
||||||
|
|
||||||
for (quint32 i = 0; i < numLinks; i++) {
|
for (int i = 0; i < numLinks; i++) {
|
||||||
FS::LinkPair pair;
|
FS::LinkPair pair;
|
||||||
in >> pair.src;
|
in >> pair.src;
|
||||||
in >> pair.dst;
|
in >> pair.dst;
|
||||||
@ -256,6 +271,7 @@ FileLinkApp::~FileLinkApp()
|
|||||||
fclose(stdout);
|
fclose(stdout);
|
||||||
fclose(stdin);
|
fclose(stdin);
|
||||||
fclose(stderr);
|
fclose(stderr);
|
||||||
|
FreeConsole();
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user