mirror of
https://github.com/nixietab/picodulce.git
synced 2025-04-07 00:48:58 +01:00
Compare commits
77 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
ba8072c669 | ||
![]() |
785e9be9f9 | ||
![]() |
0cbd000be4 | ||
![]() |
52b635285e | ||
![]() |
67a16c008a | ||
![]() |
a4bd707461 | ||
![]() |
1b27fffc96 | ||
![]() |
fade5f86b7 | ||
![]() |
823b438840 | ||
![]() |
9a8c3f44d0 | ||
![]() |
6b65fb0d1e | ||
![]() |
8247009d60 | ||
![]() |
e5c395d031 | ||
![]() |
263e6eae07 | ||
![]() |
ec99488326 | ||
![]() |
61cd427beb | ||
![]() |
cb2f5b52b3 | ||
![]() |
ba40354a5d | ||
![]() |
0c151b058e | ||
![]() |
fc7f47d273 | ||
![]() |
4f4ff35ee5 | ||
![]() |
8b9827b422 | ||
![]() |
892cbc4d07 | ||
![]() |
f61f15fe7e | ||
![]() |
d077a922c0 | ||
![]() |
9b70503d26 | ||
![]() |
ae9f25a7a8 | ||
![]() |
00ed5f97b9 | ||
![]() |
5dbbfd5d87 | ||
![]() |
37a1c5b0df | ||
![]() |
f2a1989993 | ||
![]() |
3d40ce7df3 | ||
![]() |
36ff8896ef | ||
![]() |
5f59acf0b4 | ||
![]() |
c48a193d9a | ||
![]() |
47a843c669 | ||
![]() |
52be28bb6c | ||
![]() |
6522b70066 | ||
![]() |
8d486a9af2 | ||
![]() |
97393e4ae7 | ||
![]() |
e35120bb36 | ||
![]() |
f2cfb3ceb3 | ||
![]() |
15246cd535 | ||
![]() |
3edcd10c12 | ||
![]() |
874e513b47 | ||
![]() |
a10318e00d | ||
![]() |
7608b647fe | ||
![]() |
60d16326b0 | ||
![]() |
0d300f0435 | ||
![]() |
3123ed30cf | ||
![]() |
db41858aae | ||
![]() |
8c0a794202 | ||
![]() |
514f6427ab | ||
![]() |
9920636b9c | ||
![]() |
5182e42f81 | ||
![]() |
d2f3aa6a49 | ||
![]() |
8167462838 | ||
![]() |
51dc126c8a | ||
![]() |
5202f8fb25 | ||
![]() |
46d053d952 | ||
![]() |
82f44105b5 | ||
![]() |
fd2e57ffb3 | ||
![]() |
372075132f | ||
![]() |
9c66a3eeb8 | ||
![]() |
27c75d1ef2 | ||
![]() |
18692c3abc | ||
![]() |
c4f662235e | ||
![]() |
bf30a6ecd6 | ||
![]() |
f4b0b39090 | ||
![]() |
e02ca8ab76 | ||
![]() |
b55ac71db1 | ||
![]() |
f4eac0f412 | ||
![]() |
dbca6721be | ||
![]() |
b0ec76ebc3 | ||
![]() |
0028de0d69 | ||
![]() |
17f5f2a8d0 | ||
![]() |
a19b8a1545 |
36
.github/workflows/Bleeding-Job.yaml
vendored
Normal file
36
.github/workflows/Bleeding-Job.yaml
vendored
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
name: Bleeding Update version
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- main
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
update-version:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Check out the repository
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- name: Set up Python
|
||||||
|
uses: actions/setup-python@v4
|
||||||
|
with:
|
||||||
|
python-version: '3.x'
|
||||||
|
|
||||||
|
- name: Update version.json
|
||||||
|
run: |
|
||||||
|
git fetch --prune --unshallow
|
||||||
|
commit_count=$(git rev-list --count HEAD)
|
||||||
|
version=$(jq -r '.version' version.json)
|
||||||
|
jq --arg versionBleeding "$version-$commit_count" '. + {versionBleeding: $versionBleeding}' version.json > version.tmp && mv version.tmp version.json
|
||||||
|
|
||||||
|
- name: Commit and push changes
|
||||||
|
run: |
|
||||||
|
git config --global user.name 'github-actions[bot]'
|
||||||
|
git config --global user.email 'github-actions[bot]@users.noreply.github.com'
|
||||||
|
git add version.json
|
||||||
|
git commit -m "Update version.json with commit count"
|
||||||
|
git push
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
69
.github/workflows/Build.yml
vendored
Normal file
69
.github/workflows/Build.yml
vendored
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
name: Version Change Action
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
paths:
|
||||||
|
- version.json # Trigger on changes to version.json
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
version-release:
|
||||||
|
runs-on: windows-latest # Use Windows 10 runner
|
||||||
|
|
||||||
|
if: github.actor != 'github-actions[bot]' # Only run if the actor is not the GitHub Actions bot
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- name: Set up Python
|
||||||
|
uses: actions/setup-python@v4
|
||||||
|
with:
|
||||||
|
python-version: '3.x' # Specify the Python version you need
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: |
|
||||||
|
python -m pip install --upgrade pip
|
||||||
|
pip install pyqt5 requests pywin32 pyinstaller pillow # Install specific dependencies
|
||||||
|
|
||||||
|
- name: Create actions-temp folder
|
||||||
|
run: mkdir actions-temp # Create the folder called actions-temp
|
||||||
|
|
||||||
|
- name: Download picoBuild.py script
|
||||||
|
run: curl -L -o actions-temp/picoBuild.py https://raw.githubusercontent.com/nixietab/picodulce-build-script/refs/heads/main/picoBuild.py
|
||||||
|
|
||||||
|
- name: Run picoBuild.py script
|
||||||
|
run: python actions-temp/picoBuild.py
|
||||||
|
|
||||||
|
- name: Show directory structure
|
||||||
|
run: |
|
||||||
|
dir actions-temp
|
||||||
|
dir
|
||||||
|
|
||||||
|
- name: Get version and name from version.json
|
||||||
|
id: version_info
|
||||||
|
run: |
|
||||||
|
$versionJson = Get-Content version.json | ConvertFrom-Json
|
||||||
|
echo "RELEASE_NAME=Release $($versionJson.version)" >> $env:GITHUB_ENV
|
||||||
|
echo "RELEASE_TAG=$($versionJson.version)" >> $env:GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Create GitHub Release
|
||||||
|
id: create_release
|
||||||
|
uses: actions/create-release@v1
|
||||||
|
with:
|
||||||
|
tag_name: ${{ env.RELEASE_TAG }}
|
||||||
|
release_name: ${{ env.RELEASE_NAME }}
|
||||||
|
body: "This release was created automatically by a GitHub Action."
|
||||||
|
draft: false
|
||||||
|
prerelease: false
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
|
- name: Upload Release Asset
|
||||||
|
uses: actions/upload-release-asset@v1
|
||||||
|
with:
|
||||||
|
upload_url: ${{ steps.create_release.outputs.upload_url }}
|
||||||
|
asset_path: build/2hsu.exe
|
||||||
|
asset_name: 2hsu.exe
|
||||||
|
asset_content_type: application/octet-stream
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
76
PKGBUILD
Normal file
76
PKGBUILD
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
pkgname=picodulce
|
||||||
|
pkgver=0.11.7
|
||||||
|
pkgrel=1
|
||||||
|
pkgdesc="Launcher for Minecraft based on the picomc library"
|
||||||
|
arch=('x86_64')
|
||||||
|
OPTIONS=(!strip !docs libtool emptydirs)
|
||||||
|
url="https://github.com/nixietab/picodulce"
|
||||||
|
license=('MIT') # Replace with your project's license
|
||||||
|
depends=('python' 'python-virtualenv' 'xdg-utils')
|
||||||
|
makedepends=('git')
|
||||||
|
source=("git+https://github.com/nixietab/picodulce.git")
|
||||||
|
sha256sums=('SKIP')
|
||||||
|
|
||||||
|
package() {
|
||||||
|
cd "$srcdir/$pkgname"
|
||||||
|
|
||||||
|
# Create a directory for the application in the user's home directory
|
||||||
|
install -dm755 "$pkgdir/usr/share/$pkgname"
|
||||||
|
|
||||||
|
# Copy all project files to the created directory
|
||||||
|
cp -r . "$pkgdir/usr/share/$pkgname"
|
||||||
|
|
||||||
|
# Create a virtual environment
|
||||||
|
python -m venv "$pkgdir/usr/share/$pkgname/venv"
|
||||||
|
|
||||||
|
# Activate the virtual environment and install dependencies
|
||||||
|
source "$pkgdir/usr/share/$pkgname/venv/bin/activate"
|
||||||
|
pip install -r requirements.txt
|
||||||
|
|
||||||
|
# Create a run.sh script
|
||||||
|
install -Dm755 /dev/stdin "$pkgdir/usr/share/$pkgname/run.sh" <<EOF
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
if [ ! -d "venv" ]; then
|
||||||
|
echo "venv folder does not exist. Creating virtual environment..."
|
||||||
|
python3 -m venv venv
|
||||||
|
|
||||||
|
source venv/bin/activate
|
||||||
|
|
||||||
|
echo "Installing required packages..."
|
||||||
|
pip install -r requirements.txt
|
||||||
|
else
|
||||||
|
source venv/bin/activate
|
||||||
|
fi
|
||||||
|
|
||||||
|
python picodulce.py
|
||||||
|
EOF
|
||||||
|
|
||||||
|
|
||||||
|
# Make the run.sh script executable
|
||||||
|
chmod +x "$pkgdir/usr/share/$pkgname/run.sh"
|
||||||
|
|
||||||
|
# Create a desktop entry for the application
|
||||||
|
install -Dm644 /dev/stdin "$pkgdir/usr/share/applications/$pkgname.desktop" <<EOF
|
||||||
|
[Desktop Entry]
|
||||||
|
Name=Picodulce
|
||||||
|
Exec=/usr/share/picodulce/run.sh
|
||||||
|
Icon=/usr/share/picodulce/launcher_icon.ico
|
||||||
|
Terminal=true
|
||||||
|
Type=Application
|
||||||
|
Comment=Picodulce Launcher
|
||||||
|
Categories=Game;
|
||||||
|
EOF
|
||||||
|
|
||||||
|
# Ensure the normal user has permission to write to the picodulce folder
|
||||||
|
chown -R "$USER:$USER" "$pkgdir/usr/share/$pkgname"
|
||||||
|
chmod -R u+w "$pkgdir/usr/share/$pkgname"
|
||||||
|
|
||||||
|
#Install into bin
|
||||||
|
install -Dm755 /dev/stdin "$pkgdir/usr/bin/picodulce" <<EOF
|
||||||
|
#!/bin/bash
|
||||||
|
cd /usr/share/picodulce/
|
||||||
|
exec ./run.sh
|
||||||
|
EOF
|
||||||
|
}
|
||||||
|
# vim:set ts=2 sw=2 et:
|
16
README.md
16
README.md
@ -40,7 +40,21 @@
|
|||||||
- **Custom Theme Support**: Create and apply personalized themes with ease. A dedicated repository and guide are [available to help you get started.](https://github.com/nixietab/picodulce-themes)
|
- **Custom Theme Support**: Create and apply personalized themes with ease. A dedicated repository and guide are [available to help you get started.](https://github.com/nixietab/picodulce-themes)
|
||||||
|
|
||||||
# Installation
|
# Installation
|
||||||
If you are on windows you may be more interested in a [installer](https://github.com/nixietab/2hsu/releases/download/release/2hsu.exe)
|
|
||||||
|
## Windows
|
||||||
|
For Windows systems using the [installer](https://github.com/nixietab/picodulce/releases/latest) is recommended
|
||||||
|
|
||||||
|
## Arch Linux
|
||||||
|
The package is available in the [AUR](https://aur.archlinux.org/packages/picodulce) as ```picodulce```
|
||||||
|
|
||||||
|
For installing on Arch without using an AUR helper a PKGBUILD is provided
|
||||||
|
```
|
||||||
|
git clone https://aur.archlinux.org/picodulce.git
|
||||||
|
cd picodulce
|
||||||
|
makepkg -si
|
||||||
|
```
|
||||||
|
|
||||||
|
## Other OS
|
||||||
|
|
||||||
### 1. Clone the repository
|
### 1. Clone the repository
|
||||||
|
|
||||||
|
244
authser.py
Normal file
244
authser.py
Normal file
@ -0,0 +1,244 @@
|
|||||||
|
import sys
|
||||||
|
import subprocess
|
||||||
|
import re
|
||||||
|
from PyQt5.QtWidgets import (QApplication, QDialog, QLabel, QVBoxLayout,
|
||||||
|
QPushButton, QLineEdit, QMessageBox)
|
||||||
|
from PyQt5.QtCore import QThread, pyqtSignal, Qt, QUrl, QObject
|
||||||
|
from PyQt5.QtGui import QDesktopServices
|
||||||
|
|
||||||
|
class AuthenticationParser:
|
||||||
|
@staticmethod
|
||||||
|
def clean_ansi(text):
|
||||||
|
ansi_clean = re.compile(r'\x1B(?:[@-Z\\-_]|\[[0-?]*[ -/]*[@-~])')
|
||||||
|
printable_clean = re.compile(r'[^\x20-\x7E\n]')
|
||||||
|
text = ansi_clean.sub('', text)
|
||||||
|
text = printable_clean.sub('', text)
|
||||||
|
return text.strip()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def is_auth_error(output):
|
||||||
|
cleaned_output = AuthenticationParser.clean_ansi(output)
|
||||||
|
return "AADSTS70016" in cleaned_output and "not yet been authorized" in cleaned_output
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def parse_auth_output(output):
|
||||||
|
cleaned_output = AuthenticationParser.clean_ansi(output)
|
||||||
|
if AuthenticationParser.is_auth_error(cleaned_output):
|
||||||
|
return None
|
||||||
|
|
||||||
|
pattern = r"https://[^\s]+"
|
||||||
|
code_pattern = r"code\s+([A-Z0-9]+)"
|
||||||
|
|
||||||
|
url_match = re.search(pattern, cleaned_output)
|
||||||
|
code_match = re.search(code_pattern, cleaned_output, re.IGNORECASE)
|
||||||
|
|
||||||
|
if url_match and code_match:
|
||||||
|
return {
|
||||||
|
'url': url_match.group(0),
|
||||||
|
'code': code_match.group(1)
|
||||||
|
}
|
||||||
|
return None
|
||||||
|
|
||||||
|
class AuthDialog(QDialog):
|
||||||
|
def __init__(self, url, code, parent=None, error_mode=False):
|
||||||
|
super().__init__(parent)
|
||||||
|
self.setWindowTitle("Microsoft Authentication")
|
||||||
|
self.setWindowFlags(self.windowFlags() & ~Qt.WindowContextHelpButtonHint)
|
||||||
|
self.setModal(True)
|
||||||
|
self.setup_ui(url, code, error_mode)
|
||||||
|
|
||||||
|
def setup_ui(self, url, code, error_mode):
|
||||||
|
layout = QVBoxLayout(self)
|
||||||
|
|
||||||
|
if error_mode:
|
||||||
|
error_label = QLabel("Error in Login - Please try again")
|
||||||
|
error_label.setStyleSheet("QLabel { color: red; font-weight: bold; }")
|
||||||
|
layout.addWidget(error_label)
|
||||||
|
|
||||||
|
instructions = QLabel(
|
||||||
|
"To authenticate your Microsoft Account:\n\n"
|
||||||
|
"1. Click 'Open Authentication Page' or visit:\n"
|
||||||
|
"2. Copy the code below\n"
|
||||||
|
"3. Paste the code on the Microsoft website\n"
|
||||||
|
"4. After completing authentication, click 'I've Completed Authentication'"
|
||||||
|
)
|
||||||
|
instructions.setWordWrap(True)
|
||||||
|
layout.addWidget(instructions)
|
||||||
|
|
||||||
|
url_label = QLabel(url)
|
||||||
|
url_label.setTextInteractionFlags(Qt.TextSelectableByMouse)
|
||||||
|
url_label.setWordWrap(True)
|
||||||
|
layout.addWidget(url_label)
|
||||||
|
|
||||||
|
self.code_input = QLineEdit(code)
|
||||||
|
self.code_input.setReadOnly(True)
|
||||||
|
self.code_input.setAlignment(Qt.AlignCenter)
|
||||||
|
self.code_input.setStyleSheet("""
|
||||||
|
QLineEdit {
|
||||||
|
font-size: 16pt;
|
||||||
|
font-weight: bold;
|
||||||
|
padding: 5px;
|
||||||
|
}
|
||||||
|
""")
|
||||||
|
layout.addWidget(self.code_input)
|
||||||
|
|
||||||
|
copy_button = QPushButton("Copy Code")
|
||||||
|
copy_button.clicked.connect(self.copy_code)
|
||||||
|
layout.addWidget(copy_button)
|
||||||
|
|
||||||
|
open_url_button = QPushButton("Open Authentication Page")
|
||||||
|
open_url_button.clicked.connect(lambda: self.open_url(url))
|
||||||
|
layout.addWidget(open_url_button)
|
||||||
|
|
||||||
|
continue_button = QPushButton("I've Completed Authentication")
|
||||||
|
continue_button.clicked.connect(self.accept)
|
||||||
|
layout.addWidget(continue_button)
|
||||||
|
|
||||||
|
def copy_code(self):
|
||||||
|
clipboard = QApplication.clipboard()
|
||||||
|
clipboard.setText(self.code_input.text())
|
||||||
|
|
||||||
|
def open_url(self, url):
|
||||||
|
QDesktopServices.openUrl(QUrl(url))
|
||||||
|
|
||||||
|
class AuthenticationThread(QThread):
|
||||||
|
auth_data_received = pyqtSignal(dict)
|
||||||
|
error_occurred = pyqtSignal(str)
|
||||||
|
auth_error_detected = pyqtSignal(str)
|
||||||
|
finished = pyqtSignal()
|
||||||
|
|
||||||
|
def __init__(self, account):
|
||||||
|
super().__init__()
|
||||||
|
self.account = account
|
||||||
|
self.process = None
|
||||||
|
self.is_running = True
|
||||||
|
self.current_output = ""
|
||||||
|
self.waiting_for_auth = False
|
||||||
|
|
||||||
|
def run(self):
|
||||||
|
try:
|
||||||
|
command = f'picomc account authenticate {self.account}'
|
||||||
|
|
||||||
|
self.process = subprocess.Popen(
|
||||||
|
command,
|
||||||
|
shell=True,
|
||||||
|
stdout=subprocess.PIPE,
|
||||||
|
stderr=subprocess.STDOUT,
|
||||||
|
stdin=subprocess.PIPE,
|
||||||
|
text=True,
|
||||||
|
bufsize=1,
|
||||||
|
universal_newlines=True
|
||||||
|
)
|
||||||
|
|
||||||
|
self.current_output = ""
|
||||||
|
while self.is_running and self.process.poll() is None:
|
||||||
|
line = self.process.stdout.readline()
|
||||||
|
if line:
|
||||||
|
self.current_output += line
|
||||||
|
|
||||||
|
if not self.waiting_for_auth:
|
||||||
|
parsed_data = AuthenticationParser.parse_auth_output(self.current_output)
|
||||||
|
if parsed_data:
|
||||||
|
self.auth_data_received.emit(parsed_data)
|
||||||
|
self.waiting_for_auth = True
|
||||||
|
self.current_output = ""
|
||||||
|
elif AuthenticationParser.is_auth_error(self.current_output):
|
||||||
|
self.auth_error_detected.emit(self.current_output)
|
||||||
|
self.waiting_for_auth = False
|
||||||
|
self.current_output = ""
|
||||||
|
|
||||||
|
self.process.wait()
|
||||||
|
self.finished.emit()
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
self.error_occurred.emit(str(e))
|
||||||
|
self.finished.emit()
|
||||||
|
|
||||||
|
def send_enter(self):
|
||||||
|
if self.process and self.process.poll() is None:
|
||||||
|
self.process.stdin.write("\n")
|
||||||
|
self.process.stdin.flush()
|
||||||
|
|
||||||
|
def stop(self):
|
||||||
|
self.is_running = False
|
||||||
|
if self.process:
|
||||||
|
self.process.terminate()
|
||||||
|
|
||||||
|
class MinecraftAuthenticator(QObject): # Changed to inherit from QObject
|
||||||
|
auth_finished = pyqtSignal(bool) # Add signal for completion
|
||||||
|
|
||||||
|
def __init__(self, parent=None):
|
||||||
|
super().__init__(parent)
|
||||||
|
self.auth_thread = None
|
||||||
|
self.current_auth_data = None
|
||||||
|
self.auth_dialog = None
|
||||||
|
self.success = False
|
||||||
|
|
||||||
|
def authenticate(self, username):
|
||||||
|
"""
|
||||||
|
Start the authentication process for the given username
|
||||||
|
Returns immediately, authentication result will be emitted via auth_finished signal
|
||||||
|
"""
|
||||||
|
self.success = False
|
||||||
|
self.auth_thread = AuthenticationThread(username)
|
||||||
|
self.auth_thread.auth_data_received.connect(self.show_auth_dialog)
|
||||||
|
self.auth_thread.auth_error_detected.connect(self.handle_auth_error)
|
||||||
|
self.auth_thread.error_occurred.connect(self.show_error)
|
||||||
|
self.auth_thread.finished.connect(self.on_authentication_finished)
|
||||||
|
self.auth_thread.start()
|
||||||
|
|
||||||
|
def show_auth_dialog(self, auth_data):
|
||||||
|
self.current_auth_data = auth_data
|
||||||
|
|
||||||
|
if self.auth_dialog is not None:
|
||||||
|
self.auth_dialog.close()
|
||||||
|
self.auth_dialog = None
|
||||||
|
|
||||||
|
self.auth_dialog = AuthDialog(auth_data['url'], auth_data['code'])
|
||||||
|
if self.auth_dialog.exec_() == QDialog.Accepted:
|
||||||
|
self.auth_thread.send_enter()
|
||||||
|
|
||||||
|
def handle_auth_error(self, output):
|
||||||
|
if self.current_auth_data:
|
||||||
|
if self.auth_dialog is not None:
|
||||||
|
self.auth_dialog.close()
|
||||||
|
self.auth_dialog = None
|
||||||
|
|
||||||
|
self.auth_dialog = AuthDialog(
|
||||||
|
self.current_auth_data['url'],
|
||||||
|
self.current_auth_data['code'],
|
||||||
|
error_mode=True
|
||||||
|
)
|
||||||
|
if self.auth_dialog.exec_() == QDialog.Accepted:
|
||||||
|
self.auth_thread.send_enter()
|
||||||
|
|
||||||
|
def show_error(self, error_message):
|
||||||
|
QMessageBox.critical(None, "Error", f"Authentication error: {error_message}")
|
||||||
|
self.success = False
|
||||||
|
self.auth_finished.emit(False)
|
||||||
|
|
||||||
|
def on_authentication_finished(self):
|
||||||
|
if self.auth_dialog is not None:
|
||||||
|
self.auth_dialog.close()
|
||||||
|
self.auth_dialog = None
|
||||||
|
|
||||||
|
if self.auth_thread:
|
||||||
|
self.auth_thread.stop()
|
||||||
|
self.auth_thread = None
|
||||||
|
|
||||||
|
self.success = True
|
||||||
|
self.auth_finished.emit(True)
|
||||||
|
|
||||||
|
def cleanup(self):
|
||||||
|
if self.auth_dialog is not None:
|
||||||
|
self.auth_dialog.close()
|
||||||
|
self.auth_dialog = None
|
||||||
|
|
||||||
|
if self.auth_thread and self.auth_thread.isRunning():
|
||||||
|
self.auth_thread.stop()
|
||||||
|
self.auth_thread.wait()
|
||||||
|
|
||||||
|
# Example usage
|
||||||
|
if __name__ == '__main__':
|
||||||
|
authenticator = MinecraftAuthenticator()
|
||||||
|
authenticator.authenticate("TestUser")
|
119
healthcheck.py
Normal file
119
healthcheck.py
Normal file
@ -0,0 +1,119 @@
|
|||||||
|
import os
|
||||||
|
import json
|
||||||
|
import requests
|
||||||
|
|
||||||
|
|
||||||
|
class HealthCheck:
|
||||||
|
def __init__(self):
|
||||||
|
self.config = None
|
||||||
|
|
||||||
|
def check_config_file(self):
|
||||||
|
config_path = "config.json"
|
||||||
|
default_config = {
|
||||||
|
"IsRCPenabled": False,
|
||||||
|
"CheckUpdate": False,
|
||||||
|
"IsBleeding": False,
|
||||||
|
"LastPlayed": "",
|
||||||
|
"IsFirstLaunch": True,
|
||||||
|
"Instance": "default",
|
||||||
|
"Theme": "Dark.json",
|
||||||
|
"ThemeBackground": True,
|
||||||
|
"ThemeRepository": "https://raw.githubusercontent.com/nixietab/picodulce-themes/main/repo.json",
|
||||||
|
"Locale": "en"
|
||||||
|
}
|
||||||
|
|
||||||
|
# Step 1: Check if the file exists; if not, create it with default values
|
||||||
|
if not os.path.exists(config_path):
|
||||||
|
with open(config_path, "w") as config_file:
|
||||||
|
json.dump(default_config, config_file, indent=4)
|
||||||
|
self.config = default_config
|
||||||
|
return
|
||||||
|
|
||||||
|
# Step 2: Try loading the config file, handle invalid JSON
|
||||||
|
try:
|
||||||
|
with open(config_path, "r") as config_file:
|
||||||
|
self.config = json.load(config_file)
|
||||||
|
except (json.JSONDecodeError, ValueError):
|
||||||
|
# File is corrupted, overwrite it with default configuration
|
||||||
|
with open(config_path, "w") as config_file:
|
||||||
|
json.dump(default_config, config_file, indent=4)
|
||||||
|
self.config = default_config
|
||||||
|
return
|
||||||
|
|
||||||
|
# Step 3: Check for missing keys and add defaults if necessary
|
||||||
|
updated = False
|
||||||
|
for key, value in default_config.items():
|
||||||
|
if key not in self.config: # Field is missing
|
||||||
|
self.config[key] = value
|
||||||
|
updated = True
|
||||||
|
|
||||||
|
# Step 4: Save the repaired config back to the file
|
||||||
|
if updated:
|
||||||
|
with open(config_path, "w") as config_file:
|
||||||
|
json.dump(self.config, config_file, indent=4)
|
||||||
|
|
||||||
|
def themes_integrity(self):
|
||||||
|
# Define folder and file paths
|
||||||
|
themes_folder = "themes"
|
||||||
|
dark_theme_file = os.path.join(themes_folder, "Dark.json")
|
||||||
|
native_theme_file = os.path.join(themes_folder, "Native.json")
|
||||||
|
|
||||||
|
# Define the default content for Dark.json
|
||||||
|
dark_theme_content = {
|
||||||
|
"manifest": {
|
||||||
|
"name": "Dark",
|
||||||
|
"description": "The default picodulce launcher theme",
|
||||||
|
"author": "Nixietab",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"palette": {
|
||||||
|
"Window": "#353535",
|
||||||
|
"WindowText": "#ffffff",
|
||||||
|
"Base": "#191919",
|
||||||
|
"AlternateBase": "#353535",
|
||||||
|
"ToolTipBase": "#ffffff",
|
||||||
|
"ToolTipText": "#ffffff",
|
||||||
|
"Text": "#ffffff",
|
||||||
|
"Button": "#353535",
|
||||||
|
"ButtonText": "#ffffff",
|
||||||
|
"BrightText": "#ff0000",
|
||||||
|
"Link": "#2a82da",
|
||||||
|
"Highlight": "#4bb679",
|
||||||
|
"HighlightedText": "#ffffff"
|
||||||
|
},
|
||||||
|
"background_image_base64": ""
|
||||||
|
}
|
||||||
|
|
||||||
|
# Define the default content for Native.json
|
||||||
|
native_theme_content = {
|
||||||
|
"manifest": {
|
||||||
|
"name": "Native",
|
||||||
|
"description": "The native looks of your OS",
|
||||||
|
"author": "Your Qt Style",
|
||||||
|
"license": "Any"
|
||||||
|
},
|
||||||
|
"palette": {}
|
||||||
|
}
|
||||||
|
|
||||||
|
# Step 1: Ensure the themes folder exists
|
||||||
|
if not os.path.exists(themes_folder):
|
||||||
|
print(f"Creating folder: {themes_folder}")
|
||||||
|
os.makedirs(themes_folder)
|
||||||
|
|
||||||
|
# Step 2: Ensure Dark.json exists
|
||||||
|
if not os.path.isfile(dark_theme_file):
|
||||||
|
print(f"Creating file: {dark_theme_file}")
|
||||||
|
with open(dark_theme_file, "w", encoding="utf-8") as file:
|
||||||
|
json.dump(dark_theme_content, file, indent=2)
|
||||||
|
print("Dark.json has been created successfully.")
|
||||||
|
|
||||||
|
# Step 3: Ensure Native.json exists
|
||||||
|
if not os.path.isfile(native_theme_file):
|
||||||
|
print(f"Creating file: {native_theme_file}")
|
||||||
|
with open(native_theme_file, "w", encoding="utf-8") as file:
|
||||||
|
json.dump(native_theme_content, file, indent=2)
|
||||||
|
print("Native.json has been created successfully.")
|
||||||
|
|
||||||
|
# Check if both files exist and print OK message
|
||||||
|
if os.path.isfile(dark_theme_file) and os.path.isfile(native_theme_file):
|
||||||
|
print("Theme Integrity OK")
|
797
picodulce.py
797
picodulce.py
File diff suppressed because it is too large
Load Diff
@ -2,3 +2,4 @@ picomc
|
|||||||
PyQt5
|
PyQt5
|
||||||
requests
|
requests
|
||||||
pypresence
|
pypresence
|
||||||
|
tqdm
|
||||||
|
@ -1,11 +1,14 @@
|
|||||||
{
|
{
|
||||||
"version": "0.11.2",
|
"version": "0.12.1",
|
||||||
"links": [
|
"links": [
|
||||||
"https://raw.githubusercontent.com/nixietab/picodulce/main/version.json",
|
"https://raw.githubusercontent.com/nixietab/picodulce/main/version.json",
|
||||||
"https://raw.githubusercontent.com/nixietab/picodulce/main/picodulce.py",
|
"https://raw.githubusercontent.com/nixietab/picodulce/main/picodulce.py",
|
||||||
"https://raw.githubusercontent.com/nixietab/picodulce/main/requirements.txt",
|
"https://raw.githubusercontent.com/nixietab/picodulce/main/requirements.txt",
|
||||||
"https://raw.githubusercontent.com/nixietab/picodulce/main/drums.gif",
|
"https://raw.githubusercontent.com/nixietab/picodulce/main/drums.gif",
|
||||||
"https://raw.githubusercontent.com/nixietab/picodulce/main/marroc.py",
|
"https://raw.githubusercontent.com/nixietab/picodulce/main/marroc.py",
|
||||||
"https://raw.githubusercontent.com/nixietab/picodulce/main/holiday.ico"
|
"https://raw.githubusercontent.com/nixietab/picodulce/main/holiday.ico",
|
||||||
]
|
"https://raw.githubusercontent.com/nixietab/picodulce/main/authser.py",
|
||||||
|
"https://raw.githubusercontent.com/nixietab/picodulce/main/healthcheck.py"
|
||||||
|
],
|
||||||
|
"versionBleeding": "0.12.1-188"
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user