new file: README.md
This commit is contained in:
parent
ed9ab801c1
commit
9ad5537385
170
Cargo.lock
generated
170
Cargo.lock
generated
@ -8,8 +8,10 @@ version = "0.1.0"
|
||||
dependencies = [
|
||||
"base64 0.22.1",
|
||||
"dirs",
|
||||
"env_logger",
|
||||
"futures",
|
||||
"java-locator",
|
||||
"log",
|
||||
"rand 0.9.0",
|
||||
"serde",
|
||||
"serde_json",
|
||||
@ -130,6 +132,15 @@ dependencies = [
|
||||
"zerocopy 0.7.35",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "aho-corasick"
|
||||
version = "1.1.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8e60d3430d3a69478ad0993f19238d2df97c507009a52b3c10addcd7f6bcb916"
|
||||
dependencies = [
|
||||
"memchr",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "android-activity"
|
||||
version = "0.6.0"
|
||||
@ -157,6 +168,56 @@ version = "0.2.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "fc7eb209b1518d6bb87b283c20095f5228ecda460da70b44f0802523dea6da04"
|
||||
|
||||
[[package]]
|
||||
name = "anstream"
|
||||
version = "0.6.18"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8acc5369981196006228e28809f761875c0327210a891e941f4c683b3a99529b"
|
||||
dependencies = [
|
||||
"anstyle",
|
||||
"anstyle-parse",
|
||||
"anstyle-query",
|
||||
"anstyle-wincon",
|
||||
"colorchoice",
|
||||
"is_terminal_polyfill",
|
||||
"utf8parse",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "anstyle"
|
||||
version = "1.0.10"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "55cc3b69f167a1ef2e161439aa98aed94e6028e5f9a59be9a6ffb47aef1651f9"
|
||||
|
||||
[[package]]
|
||||
name = "anstyle-parse"
|
||||
version = "0.2.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "3b2d16507662817a6a20a9ea92df6652ee4f94f914589377d69f3b21bc5798a9"
|
||||
dependencies = [
|
||||
"utf8parse",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "anstyle-query"
|
||||
version = "1.1.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "79947af37f4177cfead1110013d678905c37501914fba0efea834c3fe9a8d60c"
|
||||
dependencies = [
|
||||
"windows-sys 0.59.0",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "anstyle-wincon"
|
||||
version = "3.0.7"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "ca3534e77181a9cc07539ad51f2141fe32f6c3ffd4df76db8ad92346b003ae4e"
|
||||
dependencies = [
|
||||
"anstyle",
|
||||
"once_cell",
|
||||
"windows-sys 0.59.0",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "anyhow"
|
||||
version = "1.0.97"
|
||||
@ -608,6 +669,12 @@ dependencies = [
|
||||
"inout",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "colorchoice"
|
||||
version = "1.0.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5b63caa9aa9397e2d9480a9b13673856c78d8ac123288526c37d7839f2a86990"
|
||||
|
||||
[[package]]
|
||||
name = "combine"
|
||||
version = "4.6.7"
|
||||
@ -1013,6 +1080,29 @@ dependencies = [
|
||||
"cfg-if 1.0.0",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "env_filter"
|
||||
version = "0.1.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "186e05a59d4c50738528153b83b0b0194d3a29507dfec16eccd4b342903397d0"
|
||||
dependencies = [
|
||||
"log",
|
||||
"regex",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "env_logger"
|
||||
version = "0.11.7"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "c3716d7a920fb4fac5d84e9d4bce8ceb321e9414b4409da61b07b75c1e3d0697"
|
||||
dependencies = [
|
||||
"anstream",
|
||||
"anstyle",
|
||||
"env_filter",
|
||||
"jiff",
|
||||
"log",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "equivalent"
|
||||
version = "1.0.2"
|
||||
@ -2033,6 +2123,12 @@ dependencies = [
|
||||
"libc",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "is_terminal_polyfill"
|
||||
version = "1.70.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "7943c866cc5cd64cbc25b2e01621d07fa8eb2a1a23160ee81ce38704e97b8ecf"
|
||||
|
||||
[[package]]
|
||||
name = "isahc"
|
||||
version = "0.9.14"
|
||||
@ -2100,6 +2196,30 @@ dependencies = [
|
||||
"system-deps",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "jiff"
|
||||
version = "0.2.4"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "d699bc6dfc879fb1bf9bdff0d4c56f0884fc6f0d0eb0fba397a6d00cd9a6b85e"
|
||||
dependencies = [
|
||||
"jiff-static",
|
||||
"log",
|
||||
"portable-atomic",
|
||||
"portable-atomic-util",
|
||||
"serde",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "jiff-static"
|
||||
version = "0.2.4"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8d16e75759ee0aa64c57a56acbf43916987b20c77373cb7e808979e02b93c9f9"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
"syn 2.0.100",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "jni"
|
||||
version = "0.21.1"
|
||||
@ -3127,6 +3247,21 @@ dependencies = [
|
||||
"universal-hash",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "portable-atomic"
|
||||
version = "1.11.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "350e9b48cbc6b0e028b0473b114454c6316e57336ee184ceab6e53f72c178b3e"
|
||||
|
||||
[[package]]
|
||||
name = "portable-atomic-util"
|
||||
version = "0.2.4"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "d8a2f0d8d040d7848a709caf78912debcc3f33ee4b3cac47d73d1e1069e83507"
|
||||
dependencies = [
|
||||
"portable-atomic",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "powerfmt"
|
||||
version = "0.2.0"
|
||||
@ -3370,6 +3505,35 @@ dependencies = [
|
||||
"thiserror 2.0.12",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "regex"
|
||||
version = "1.11.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "b544ef1b4eac5dc2db33ea63606ae9ffcfac26c1416a2806ae0bf5f56b201191"
|
||||
dependencies = [
|
||||
"aho-corasick",
|
||||
"memchr",
|
||||
"regex-automata",
|
||||
"regex-syntax",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "regex-automata"
|
||||
version = "0.4.9"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "809e8dc61f6de73b46c85f4c96486310fe304c434cfa43669d7b40f711150908"
|
||||
dependencies = [
|
||||
"aho-corasick",
|
||||
"memchr",
|
||||
"regex-syntax",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "regex-syntax"
|
||||
version = "0.8.5"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "2b15c43186be67a4fd63bee50d0303afffcef381492ebe2c5d87f324e1b8815c"
|
||||
|
||||
[[package]]
|
||||
name = "rustc-demangle"
|
||||
version = "0.1.24"
|
||||
@ -4424,6 +4588,12 @@ version = "1.0.4"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "b6c140620e7ffbb22c2dee59cafe6084a59b5ffc27a8859a5f0d494b5d52b6be"
|
||||
|
||||
[[package]]
|
||||
name = "utf8parse"
|
||||
version = "0.2.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "06abde3611657adf66d383f00b093d7faecc7fa57071cce2578660c9f1010821"
|
||||
|
||||
[[package]]
|
||||
name = "value-bag"
|
||||
version = "1.10.0"
|
||||
|
@ -12,8 +12,10 @@ dirs = "6.0.0"
|
||||
rand = "0.9.0"
|
||||
serde = { version = "1.0.219", features = ["derive"] }
|
||||
serde_json = "1.0"
|
||||
toml = "0.8.20"
|
||||
surf = { version = "2.3.2", features = ["hyper-client"] }
|
||||
base64 = "0.22.1"
|
||||
zip-extract = "0.2.1"
|
||||
java-locator = "0.1.9"
|
||||
java-locator = "0.1.9"
|
||||
log = "0.4.26"
|
||||
env_logger = "0.11.7"
|
||||
toml = "0.8.20"
|
49
README.md
Normal file
49
README.md
Normal file
@ -0,0 +1,49 @@
|
||||
# XCraft - A Modern Minecraft Launcher
|
||||
|
||||
XCraft is a custom Minecraft launcher written in Rust, designed to provide enhanced flexibility and customization. It supports a custom online mode, skin and cape editing, and seamless integration with MultiMC instances.
|
||||
|
||||
## Features
|
||||
|
||||
- **Custom Online Mode**: Authenticate and play on your own custom Minecraft servers.
|
||||
- **Skin & Cape Editing**: Easily customize your in-game appearance.
|
||||
- **MultiMC Support**: Load and manage MultiMC instances directly from XCraft.
|
||||
- **One-Click Mod Installation**: Install Forge, Fabric, and Omniarchive versions effortlessly.
|
||||
- **Portable**: You can run launcher in portable mode from flash drive to play your lovely game everywhere you want.
|
||||
|
||||
## Installation
|
||||
|
||||
### Download Stable Build
|
||||
You can download the latest stable build of XCraft from our Jenkins CI server:
|
||||
[Download from Jenkins](https://jenkins.awain.net/job/XCraft/lastStableBuild/).
|
||||
|
||||
### Build from Source
|
||||
|
||||
```sh
|
||||
# Clone the repository
|
||||
git clone https://github.com/yourusername/XCraft.git
|
||||
cd XCraft
|
||||
|
||||
# Build the launcher
|
||||
cargo build --release
|
||||
|
||||
# Run the launcher
|
||||
./target/release/xcraft
|
||||
```
|
||||
|
||||
## Usage
|
||||
1. Launch XCraft.
|
||||
2. Configure your custom authentication settings.
|
||||
3. Manage skins and capes within the built-in editor.
|
||||
4. Load your MultiMC instances for easy access.
|
||||
5. Enjoy a seamless Minecraft experience!
|
||||
|
||||
## For Server Administrators
|
||||
To make your Minecraft server compatible with XCraft, install the **XCraft-Auth** Spigot plugin. This plugin enables custom authentication and ensures seamless integration with XCraft's custom online mode.
|
||||
|
||||
## Roadmap
|
||||
- [ ] Add support for more mod loaders (Fabric, Forge, etc.)
|
||||
- [ ] Enhance logging and debugging features
|
||||
- [ ] Cross-platform support improvements
|
||||
|
||||
## License
|
||||
This project is licensed under the MIT License.
|
@ -2,14 +2,37 @@ use std::path::PathBuf;
|
||||
|
||||
use serde::{Deserialize, Serialize};
|
||||
|
||||
#[derive(Serialize, Deserialize)]
|
||||
pub struct LauncherCredentials {
|
||||
pub uuid: String,
|
||||
pub username: String,
|
||||
pub password: String
|
||||
}
|
||||
|
||||
#[derive(Serialize, Deserialize)]
|
||||
pub struct LauncherServer {
|
||||
pub domain: String,
|
||||
pub port: u16,
|
||||
pub session_server_port: u16,
|
||||
pub credentials: LauncherCredentials
|
||||
}
|
||||
|
||||
#[derive(Default, Serialize, Deserialize)]
|
||||
#[derive(Serialize, Deserialize)]
|
||||
pub struct LauncherConfig {
|
||||
is_portable: bool,
|
||||
user_name: String,
|
||||
pub user_secret: String,
|
||||
pub java_path: String
|
||||
pub java_path: String,
|
||||
pub show_alpha: bool,
|
||||
pub show_beta: bool,
|
||||
pub show_snapshots: bool,
|
||||
pub ram_amount: u32,
|
||||
servers: Vec<LauncherServer>
|
||||
}
|
||||
|
||||
impl Default for LauncherConfig {
|
||||
fn default() -> Self {
|
||||
Self { is_portable: Default::default(), user_name: Default::default(), java_path: "java".to_string(), show_alpha: true, show_beta: true, show_snapshots: false, ram_amount: 1024, servers: Default::default() }
|
||||
}
|
||||
}
|
||||
|
||||
impl LauncherConfig {
|
||||
@ -37,6 +60,10 @@ impl LauncherConfig {
|
||||
pub fn set_username(&mut self, user_name: String) {
|
||||
self.user_name = user_name;
|
||||
}
|
||||
|
||||
pub fn add_server(&mut self, server: LauncherServer) {
|
||||
self.servers.push(server);
|
||||
}
|
||||
}
|
||||
|
||||
pub fn get_relative_launcher_dir() -> PathBuf {
|
||||
|
@ -1,11 +1,14 @@
|
||||
use core::str;
|
||||
use std::io::Cursor;
|
||||
use base64::{encode, Engine};
|
||||
use base64::Engine;
|
||||
use base64::prelude::BASE64_STANDARD;
|
||||
use tokio::fs::File;
|
||||
use tokio::process::Command;
|
||||
use tokio::sync::mpsc;
|
||||
use tokio::sync::mpsc::UnboundedSender;
|
||||
use crate::config::{LauncherCredentials, LauncherServer};
|
||||
use crate::minecraft;
|
||||
use crate::minecraft::session::SignUpResponse;
|
||||
use crate::minecraft::versions::Version;
|
||||
use crate::{config::LauncherConfig, minecraft::versions::VersionConfig, util};
|
||||
|
||||
@ -56,10 +59,54 @@ impl Launcher {
|
||||
pub fn init_config(&mut self, user_name: String) {
|
||||
self.load_config();
|
||||
self.config.set_username(user_name);
|
||||
self.config.user_secret = crate::util::random_string(32);
|
||||
self.save_config();
|
||||
}
|
||||
|
||||
fn save_server_info(&mut self, uuid: String, username: String, password: String, domain: String, session_server_port: u16, server_port: u16) -> (bool, &str) {
|
||||
self.config.add_server(LauncherServer {
|
||||
domain,
|
||||
port: server_port,
|
||||
session_server_port,
|
||||
credentials: LauncherCredentials {
|
||||
uuid,
|
||||
username,
|
||||
password
|
||||
}
|
||||
});
|
||||
self.save_config();
|
||||
(true, "You are successfully registered")
|
||||
}
|
||||
|
||||
pub async fn register_user_server(&mut self, server: String, username: String, password: String) -> (bool, &str) {
|
||||
let mut session_server_port: u16 = 8999;
|
||||
let mut server_port: u16 = 25565;
|
||||
let mut domain = server.clone();
|
||||
if let Some(index) = server.find("#") {
|
||||
let (a,b) = server.split_at(index+1);
|
||||
session_server_port = b.parse().unwrap();
|
||||
domain = a[..a.len()-1].to_string();
|
||||
}
|
||||
|
||||
if let Some(index) = domain.find(":") {
|
||||
let dmc = domain.clone();
|
||||
let (a,b) = dmc.split_at(index+1);
|
||||
domain = a[..a.len()-1].to_string();
|
||||
server_port = b.parse().unwrap();
|
||||
}
|
||||
|
||||
println!("Server information: {}:{} session={}", domain, server_port, session_server_port);
|
||||
|
||||
match minecraft::session::try_signup(domain.clone(), session_server_port, username.clone(), password.clone()).await {
|
||||
Ok(status) => match status {
|
||||
SignUpResponse::ServerError => (false, "Internal server error"),
|
||||
SignUpResponse::BadCredentials => (false, "Username or password is not valid"),
|
||||
SignUpResponse::UserAlreadyExists => (false, "User already exists"),
|
||||
SignUpResponse::Registered(uuid) => self.save_server_info(uuid, username, password, domain, session_server_port, server_port)
|
||||
}
|
||||
Err(_e) => (false, "Internal server error")
|
||||
}
|
||||
}
|
||||
|
||||
pub fn get_instances_list(&self) -> Vec<(String, String, String)> {
|
||||
let mut v = Vec::new();
|
||||
let mut instances = self.config.launcher_dir();
|
||||
@ -87,7 +134,7 @@ impl Launcher {
|
||||
v
|
||||
}
|
||||
|
||||
pub async fn launch_instance(&self, instance_name: String) {
|
||||
pub async fn launch_instance(&self, instance_name: String, username: String, uuid: String, token: String) {
|
||||
let mut instances = self.config.launcher_dir();
|
||||
instances.push("instances");
|
||||
instances.push(&instance_name);
|
||||
@ -145,7 +192,7 @@ impl Launcher {
|
||||
|
||||
let mut assets_dir = self.config.launcher_dir();
|
||||
assets_dir.push("assets");
|
||||
cmd.args(&["--username", self.config.user_name(), "--version", &instance_name, "--gameDir", game_dir.to_str().unwrap(), "--assetsDir", assets_dir.to_str().unwrap(), "--assetIndex", &config.assetIndex.id, "--uuid", "51820246d9fe372b81592602a5239ad9", "--accessToken", "51820246d9fe372b81592602a5239ad9", "--userProperties", "{}", "--userType", "mojang", "--width", "925", "--height", "530"]);
|
||||
cmd.args(&["--username", &username, "--version", &instance_name, "--gameDir", game_dir.to_str().unwrap(), "--assetsDir", assets_dir.to_str().unwrap(), "--assetIndex", &config.assetIndex.id, "--uuid", &uuid, "--accessToken", &token, "--userProperties", "{}", "--userType", "mojang", "--width", "925", "--height", "530"]);
|
||||
cmd.spawn();
|
||||
}
|
||||
}
|
||||
@ -269,7 +316,7 @@ impl Launcher {
|
||||
pub fn init_dirs(&self) {
|
||||
let root = self.config.launcher_dir();
|
||||
std::fs::create_dir_all(&root);
|
||||
// instances assets libraries config.toml servers credentials
|
||||
// instances assets libraries config.toml
|
||||
let mut instances = root.clone();
|
||||
instances.push("instances");
|
||||
|
||||
@ -279,16 +326,8 @@ impl Launcher {
|
||||
let mut libraries = root.clone();
|
||||
libraries.push("libraries");
|
||||
|
||||
let mut servers = root.clone();
|
||||
servers.push("servers");
|
||||
|
||||
let mut credentials = root.clone();
|
||||
credentials.push("credentials");
|
||||
|
||||
std::fs::create_dir_all(&instances);
|
||||
std::fs::create_dir_all(&assets);
|
||||
std::fs::create_dir_all(&libraries);
|
||||
std::fs::create_dir_all(&servers);
|
||||
std::fs::create_dir_all(&credentials);
|
||||
}
|
||||
}
|
21
src/main.rs
21
src/main.rs
@ -111,7 +111,16 @@ async fn main() {
|
||||
}
|
||||
"fetch_official_versions" => {
|
||||
if let Ok(versions) = crate::minecraft::versions::fetch_versions_list().await {
|
||||
let versions: Vec<String> = versions.versions.iter().map(|t| t.id.clone()).collect();
|
||||
let versions: Vec<String> = versions.versions.iter().filter(|t| {
|
||||
if !launcher.config.show_alpha && t.r#type == "old_alpha" {
|
||||
return false;
|
||||
} else if !launcher.config.show_beta && t.r#type == "old_beta" {
|
||||
return false;
|
||||
} else if !launcher.config.show_snapshots && t.r#type == "snapshot" {
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}).map(|t| t.id.clone()).collect();
|
||||
responder.respond(Response::new(serde_json::to_vec(&UIMessage { params: [ vec!["set_downloadable_versions".to_string()], versions ].concat() }).unwrap()));
|
||||
} else {
|
||||
responder.respond(Response::new(serde_json::to_vec(&UIMessage { params: Vec::new() }).unwrap()));
|
||||
@ -159,7 +168,7 @@ async fn main() {
|
||||
}
|
||||
"run_instance" => {
|
||||
let instance_name = params.unwrap().params[0].clone();
|
||||
launcher.launch_instance(instance_name).await;
|
||||
launcher.launch_instance(instance_name, launcher.config.user_name().to_string(), util::random_string(32), util::random_string(32)).await;
|
||||
}
|
||||
"locate_java" => {
|
||||
if let Ok(java_path) = java_locator::locate_file("java.exe") {
|
||||
@ -169,6 +178,14 @@ async fn main() {
|
||||
// todo: implement error notifications
|
||||
}
|
||||
}
|
||||
"add_server" => {
|
||||
let params = ¶ms.unwrap().params;
|
||||
let (status, msg) = launcher.register_user_server(params[0].clone(), params[1].clone(), params[2].clone()).await;
|
||||
responder.respond(Response::new(serde_json::to_vec(&UIMessage { params: vec!["add_server_response".to_string(), status.to_string(), msg.to_string()] }).unwrap()));
|
||||
}
|
||||
"fetch_settings" => {
|
||||
responder.respond(Response::new(serde_json::to_vec(&UIMessage { params: vec!["fetch_settings_response".to_string(), launcher.config.show_alpha.to_string(), launcher.config.show_beta.to_string(), launcher.config.show_snapshots.to_string(), launcher.config.java_path.clone(), launcher.config.ram_amount.to_string()] }).unwrap()));
|
||||
}
|
||||
_ => {}
|
||||
}
|
||||
}
|
||||
|
@ -152,6 +152,49 @@ pub mod versions {
|
||||
}
|
||||
}
|
||||
|
||||
pub mod session {
|
||||
use std::error::Error;
|
||||
|
||||
use serde::{Deserialize, Serialize};
|
||||
|
||||
#[derive(Serialize)]
|
||||
struct SignUpRequest {
|
||||
username: String,
|
||||
password: String,
|
||||
}
|
||||
|
||||
pub enum SignUpResponse {
|
||||
Registered(String),
|
||||
BadCredentials,
|
||||
UserAlreadyExists,
|
||||
ServerError
|
||||
}
|
||||
|
||||
#[derive(Deserialize)]
|
||||
struct ResponseUUID {
|
||||
uuid: String
|
||||
}
|
||||
|
||||
pub async fn try_signup(server_domain: String, port: u16, username: String, password: String) -> Result<SignUpResponse, Box<dyn Error + Send + Sync>> {
|
||||
let request = SignUpRequest { username, password };
|
||||
let mut r = surf::post(["http://".to_string(), server_domain, ":".to_string(), port.to_string(), "/api/register".to_string()].concat())
|
||||
.body_json(&request)
|
||||
.unwrap()
|
||||
.await?;
|
||||
|
||||
let b= r.body_bytes().await.unwrap();
|
||||
match r.status() {
|
||||
surf::StatusCode::BadRequest => Ok(SignUpResponse::BadCredentials),
|
||||
surf::StatusCode::Conflict => Ok(SignUpResponse::UserAlreadyExists),
|
||||
surf::StatusCode::Ok => {
|
||||
let response: ResponseUUID = serde_json::from_slice(&b).unwrap();
|
||||
return Ok(SignUpResponse::Registered(response.uuid))
|
||||
},
|
||||
_ => Ok(SignUpResponse::ServerError)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub mod assets {
|
||||
use std::{collections::HashMap, error::Error, path::PathBuf};
|
||||
use serde::{Deserialize, Serialize};
|
||||
|
@ -128,19 +128,17 @@
|
||||
|
||||
<div class="flex flex-col justify-center items-center">
|
||||
|
||||
<div role="alert" class="mt-5 rounded-xl border border-gray-100 bg-white p-4 w-96 hidden">
|
||||
<div id="popup" role="alert" class="mt-5 rounded-xl border border-gray-100 bg-white p-4 w-96 hidden">
|
||||
<div class="flex items-start gap-4">
|
||||
<span class="text-green-600">
|
||||
<i class="fa-solid fa-info"></i>
|
||||
</span>
|
||||
|
||||
<div class="flex-1">
|
||||
<strong class="block font-medium text-gray-900"> Changes saved </strong>
|
||||
|
||||
<p class="mt-1 text-sm text-gray-700">Your product changes have been saved.</p>
|
||||
<p id="popup_text" class="mt-1 text-sm text-gray-700">Your product changes have been saved.</p>
|
||||
</div>
|
||||
|
||||
<button class="text-gray-500 transition hover:text-gray-600">
|
||||
<button onClick="dismissPopup()" class="text-gray-500 transition hover:text-gray-600">
|
||||
<span class="sr-only">Dismiss popup</span>
|
||||
|
||||
<svg
|
||||
@ -269,33 +267,33 @@
|
||||
</div>
|
||||
|
||||
<!-- Sections -->
|
||||
<div id="launcher" class="settings-tab mt-4 hidden">
|
||||
<div id="launcher-settings" class="settings-tab mt-4">
|
||||
<h3 class="text-lg font-semibold text-gray-700"><i class="fa-solid fa-rocket"></i> Launcher Settings</h3>
|
||||
|
||||
<div class="mt-4 flex justify-between items-center">
|
||||
<label class="inline-flex items-center cursor-pointer">
|
||||
<input type="checkbox" value="" class="sr-only peer">
|
||||
<input id="show-beta" type="checkbox" value="" class="sr-only peer">
|
||||
<div class="relative w-11 h-6 bg-gray-200 peer-focus:outline-none peer-focus:ring-4 peer-focus:ring-green-300 dark:peer-focus:ring-green-800 rounded-full peer dark:bg-gray-700 peer-checked:after:translate-x-full rtl:peer-checked:after:-translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:start-[2px] after:bg-white after:border-gray-300 after:border after:rounded-full after:h-5 after:w-5 after:transition-all dark:border-gray-600 peer-checked:bg-green-600 dark:peer-checked:bg-green-600"></div>
|
||||
<span class="ms-3 text-sm font-medium text-gray-900 dark:text-gray-300">Show beta versions</span>
|
||||
</label>
|
||||
</div>
|
||||
<div class="mt-4 flex justify-between items-center">
|
||||
<label class="inline-flex items-center cursor-pointer">
|
||||
<input type="checkbox" value="" class="sr-only peer">
|
||||
<input id="show-alpha" type="checkbox" value="" class="sr-only peer">
|
||||
<div class="relative w-11 h-6 bg-gray-200 peer-focus:outline-none peer-focus:ring-4 peer-focus:ring-green-300 dark:peer-focus:ring-green-800 rounded-full peer dark:bg-gray-700 peer-checked:after:translate-x-full rtl:peer-checked:after:-translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:start-[2px] after:bg-white after:border-gray-300 after:border after:rounded-full after:h-5 after:w-5 after:transition-all dark:border-gray-600 peer-checked:bg-green-600 dark:peer-checked:bg-green-600"></div>
|
||||
<span class="ms-3 text-sm font-medium text-gray-900 dark:text-gray-300">Show alpha versions</span>
|
||||
</label>
|
||||
</div>
|
||||
<div class="mt-4 flex justify-between items-center">
|
||||
<label class="inline-flex items-center cursor-pointer">
|
||||
<input type="checkbox" value="" class="sr-only peer">
|
||||
<input id="show-snapshots" type="checkbox" value="" class="sr-only peer">
|
||||
<div class="relative w-11 h-6 bg-gray-200 peer-focus:outline-none peer-focus:ring-4 peer-focus:ring-green-300 dark:peer-focus:ring-green-800 rounded-full peer dark:bg-gray-700 peer-checked:after:translate-x-full rtl:peer-checked:after:-translate-x-full peer-checked:after:border-white after:content-[''] after:absolute after:top-[2px] after:start-[2px] after:bg-white after:border-gray-300 after:border after:rounded-full after:h-5 after:w-5 after:transition-all dark:border-gray-600 peer-checked:bg-green-600 dark:peer-checked:bg-green-600"></div>
|
||||
<span class="ms-3 text-sm font-medium text-gray-900 dark:text-gray-300">Show snapshots</span>
|
||||
</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div id="java" class="settings-tab mt-4">
|
||||
<div id="java-settings" class="settings-tab mt-4 hidden">
|
||||
<h3 class="text-lg font-semibold text-gray-700"><i class="fa-solid fa-mug-hot"></i> Java Settings</h3>
|
||||
<div class="mt-4 flex items-center border border-gray-300 rounded-lg overflow-hidden">
|
||||
<input id="java-path" type="text" placeholder="Enter Java Path"
|
||||
@ -305,14 +303,14 @@
|
||||
Auto
|
||||
</button>
|
||||
</div>
|
||||
<label for="number-input" class="block mb-2 text-sm font-medium text-gray-800 dark:text-white">Allocate memory (MB):</label>
|
||||
<input type="number" id="number-input" min="1024" step="1" aria-describedby="helper-text-explanation" class="bg-white border border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-green-500 focus:border-green-500 block w-full p-2.5" placeholder="1024" required />
|
||||
<label for="ram-input" class="block mb-2 text-sm font-medium text-gray-800 dark:text-white">Allocate memory (MB):</label>
|
||||
<input type="number" id="ram-input" min="1024" step="1" aria-describedby="helper-text-explanation" class="bg-white border border-gray-300 text-gray-900 text-sm rounded-lg focus:ring-green-500 focus:border-green-500 block w-full p-2.5" placeholder="1024" value="1024" required />
|
||||
</div>
|
||||
|
||||
|
||||
|
||||
<button
|
||||
class="w-full mt-6 bg-green-500 hover:bg-green-600 text-white font-bold py-2 px-4 rounded transition">
|
||||
onClick="saveSettings()" class="w-full mt-6 bg-green-500 hover:bg-green-600 text-white font-bold py-2 px-4 rounded transition">
|
||||
<i class="fa-solid fa-floppy-disk"></i> Save
|
||||
</button>
|
||||
|
||||
@ -380,15 +378,35 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div id="add-server-section" class="xsection bg-white shadow-lg rounded-xl p-6 w-96 text-center hidden">
|
||||
<h2 class="text-2xl font-semibold text-gray-700">Add Server</h2>
|
||||
<input id="server_address" type="text" placeholder="127.0.0.1:25565#8999" class="mt-4 w-full px-4 py-2 border rounded-lg focus:outline-none focus:ring-2 focus:ring-green-500">
|
||||
<p class="mt-2 text-sm text-green-500 text-left">XCraft servers are using their own session servers, to specify its port, use #PORT, or don't if it's 8999</p>
|
||||
|
||||
<input id="server_username" type="text" placeholder="Username" class="mt-4 w-full px-4 py-2 border rounded-lg focus:outline-none focus:ring-2 focus:ring-green-500"> <!-- bg-red-50 border border-red-500 text-red-900 placeholder-red-700 focus:ring-2 focus:ring-red-500 -->
|
||||
<p class="mt-2 text-sm text-red-600 dark:text-red-500 text-left hidden">Username has invalid characters or too short</p>
|
||||
|
||||
<input id="server_password" type="password" placeholder="Password" class="mt-4 w-full px-4 py-2 border rounded-lg focus:outline-none focus:ring-2 focus:ring-green-500">
|
||||
<!-- Sign In Button -->
|
||||
<button
|
||||
onclick="addServerInstance()"
|
||||
class="mt-4 w-full bg-green-500 hover:bg-green-600 text-white font-bold py-2 px-4 rounded transition">
|
||||
Add
|
||||
</button>
|
||||
|
||||
<!-- Footer -->
|
||||
<p class="mt-4 text-sm text-gray-500">alterdekim</p>
|
||||
</div>
|
||||
|
||||
<div id="servers-section" class="xsection grid grid-cols-3 gap-4 p-6 w-fill hidden">
|
||||
<div class="bg-white cursor-pointer hover:bg-green-500 hover:text-white shadow-lg rounded-xl w-48 h-24 flex justify-center items-center">
|
||||
<!--<div class="bg-white cursor-pointer hover:bg-green-500 hover:text-white shadow-lg rounded-xl w-48 h-24 flex justify-center items-center">
|
||||
<img src="https://eu.mc-api.net/v3/server/favicon/mc.hypixel.net" class="w-12 h-12 rounded-full">
|
||||
<div class="h-fill ms-2">
|
||||
<h2 class="text-lg font-semibold">Hypixel</h2>
|
||||
<h2 class="text-sm font-semibold">mc.hypixel.net</h2>
|
||||
</div>
|
||||
</div>
|
||||
<div class="bg-white shadow-lg rounded-xl w-48 h-24 flex justify-center items-center text-3xl text-green-500 cursor-pointer hover:bg-green-500 hover:text-white"><i class="fa-solid fa-plus"></i></div>
|
||||
</div>-->
|
||||
<div onClick="addServer()" class="bg-white shadow-lg rounded-xl w-48 h-24 flex justify-center items-center text-3xl text-green-500 cursor-pointer hover:bg-green-500 hover:text-white"><i class="fa-solid fa-plus"></i></div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@ -424,10 +442,49 @@
|
||||
} else if( params[i] == "locate_java" ) {
|
||||
setJavaPath(params[i+1]);
|
||||
break;
|
||||
} else if( params[i] == "add_server_response" ) {
|
||||
addServerResponse(params[i+1], params[i+2]);
|
||||
break;
|
||||
} else if( params[i] == "fetch_settings_response" ) {
|
||||
setSettings(params.slice(i+1));
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function setSettings(params) {
|
||||
$("#show-alpha").prop('checked', (params[0] === 'true'));
|
||||
$("#show-beta").prop('checked', (params[1] === 'true'));
|
||||
$("#show-snapshots").prop('checked', (params[2] === 'true'));
|
||||
$("#java-path").val(params[3]);
|
||||
$("#ram-input").val(params[4]);
|
||||
}
|
||||
|
||||
function dismissPopup() {
|
||||
$("#popup").addClass("hidden");
|
||||
}
|
||||
|
||||
function showPopup(text) {
|
||||
$("#popup_text").html(text);
|
||||
$("#popup").removeClass("hidden");
|
||||
}
|
||||
|
||||
function addServerResponse(status, msg) {
|
||||
showPopup(msg);
|
||||
}
|
||||
|
||||
function addServerInstance() {
|
||||
let server = $("#server_address").val();
|
||||
let username = $("#server_username").val();
|
||||
let password = $("#server_password").val();
|
||||
|
||||
$.post({url: "add_server", data: JSON.stringify({ params: [server, username, password] }) }, processParams);
|
||||
}
|
||||
|
||||
function addServer() {
|
||||
showSection(undefined, "add-server");
|
||||
}
|
||||
|
||||
function setJavaPath(javaPath) {
|
||||
$("#java-path").val(javaPath);
|
||||
}
|
||||
@ -517,7 +574,7 @@
|
||||
|
||||
$( document ).ready(async function() {
|
||||
$.get("check_installation", processParams);
|
||||
|
||||
$.get("fetch_settings", processParams);
|
||||
setInterval(function() {
|
||||
if( !$("#loading-section").hasClass("hidden") ) {
|
||||
$.get("check_download_status", processParams);
|
||||
|
Loading…
x
Reference in New Issue
Block a user