MCModpackManager/mmm/src/main.rs

433 lines
15 KiB
Rust
Raw Normal View History

use std::fmt::format;
2024-08-21 23:03:56 +01:00
use std::path::PathBuf;
use std::str::FromStr;
2024-08-21 23:03:56 +01:00
use iced::widget::{
button, checkbox, column, container, horizontal_rule, pick_list, progress_bar, row, scrollable,
slider, text, text_input, toggler, vertical_rule, vertical_space, Column,
2024-08-21 23:03:56 +01:00
};
use iced::{executor, Application, Command, Executor};
2024-08-21 23:03:56 +01:00
use iced::{Alignment, Element, Length, Sandbox, Settings, Theme};
use mcmpmgr::profiles::{self, Profile};
use mcmpmgr::providers::DownloadSide;
2024-08-21 23:03:56 +01:00
pub fn main() -> iced::Result {
ManagerGUI::run(Settings::default())
}
#[derive(Default)]
struct ManagerGUI {
theme: Theme,
selected_profile: Option<String>,
userdata: profiles::Data,
userdata_load_error: Option<String>,
2024-08-21 23:03:56 +01:00
current_view: ManagerView,
previous_view: ManagerView,
profile_edit_settings: ProfileSettings,
profile_save_error: Option<String>,
current_install_status: ProfileInstallStatus
2024-08-21 23:03:56 +01:00
}
#[derive(Debug, Clone)]
/// The current application view
enum ManagerView {
ProfileSelect,
ProfileView { profile: String },
AddProfile,
EditProfile { profile: String },
}
#[derive(Debug, Clone)]
/// The current application view
struct ProfileSettings {
name: String,
mods_dir: Option<PathBuf>,
pack_source: String,
side: DownloadSide,
2024-08-21 23:03:56 +01:00
}
impl Default for ProfileSettings {
fn default() -> Self {
Self {
name: Default::default(),
mods_dir: Default::default(),
pack_source: Default::default(),
side: DownloadSide::Client,
2024-08-21 23:03:56 +01:00
}
}
}
impl TryFrom<ProfileSettings> for profiles::Profile {
type Error = String;
fn try_from(value: ProfileSettings) -> Result<Self, Self::Error> {
let mods_dir = value
.mods_dir
.ok_or(format!("A mods directory is required"))?;
let pack_source = value.pack_source;
Ok(profiles::Profile::new(
&mods_dir,
profiles::PackSource::from_str(&pack_source)?,
value.side,
))
}
}
2024-08-21 23:03:56 +01:00
impl Default for ManagerView {
fn default() -> Self {
Self::ProfileSelect
}
}
#[derive(Debug, Clone)]
enum Message {
SwitchView(ManagerView),
BrowseModsDir,
EditProfileName(String),
EditPackSource(String),
SaveProfile,
DeleteProfile(String),
InstallProfile(String),
ProfileInstalled(ProfileInstallStatus)
}
#[derive(Debug, Clone)]
enum ProfileInstallStatus {
NotStarted,
Installing,
Success,
Error(String)
}
impl Default for ProfileInstallStatus {
fn default() -> Self {
Self::NotStarted
}
2024-08-21 23:03:56 +01:00
}
impl Application for ManagerGUI {
2024-08-21 23:03:56 +01:00
type Message = Message;
type Executor = executor::Default;
type Theme = Theme;
type Flags = ();
2024-08-21 23:03:56 +01:00
fn new(_flags: Self::Flags) -> (Self, Command<Self::Message>) {
2024-08-21 23:03:56 +01:00
let mut gui = ManagerGUI::default();
gui.theme = Theme::GruvboxDark;
let loaded_userdata = profiles::Data::load();
match loaded_userdata {
Ok(userdata) => {
gui.userdata = userdata;
}
Err(err) => {
gui.userdata_load_error = Some(err.to_string());
}
};
(gui, Command::none())
2024-08-21 23:03:56 +01:00
}
fn title(&self) -> String {
String::from("Minecraft Modpack Manager")
}
fn update(&mut self, message: Message) -> Command<Message> {
2024-08-21 23:03:56 +01:00
match message {
Message::SwitchView(view) => {
match &view {
ManagerView::AddProfile => {
self.profile_save_error = None;
self.profile_edit_settings = ProfileSettings::default();
}
ManagerView::ProfileSelect => {
let loaded_userdata = profiles::Data::load();
match loaded_userdata {
Ok(userdata) => {
self.userdata = userdata;
self.userdata_load_error = None;
}
Err(err) => {
self.userdata_load_error = Some(err.to_string());
}
};
}
2024-08-24 03:22:39 +01:00
ManagerView::EditProfile { profile } => {
let loaded_profile = self.userdata.get_profile(profile);
self.profile_edit_settings.name = profile.trim().into();
if let Some(loaded_profile) = loaded_profile {
self.profile_edit_settings.name = profile.into();
self.profile_edit_settings.mods_dir =
Some(loaded_profile.mods_folder.clone());
self.profile_edit_settings.pack_source =
loaded_profile.pack_source.to_string();
self.profile_edit_settings.side = loaded_profile.side;
} else {
eprintln!("Failed to load existing profile data for {profile}");
}
}
2024-08-21 23:03:56 +01:00
_ => {}
};
self.current_view = view;
Command::none()
2024-08-21 23:03:56 +01:00
}
Message::BrowseModsDir => {
self.profile_edit_settings.mods_dir = rfd::FileDialog::new()
.set_title("Select your mods folder")
.pick_folder();
Command::none()
}
Message::EditProfileName(name) => {
self.profile_edit_settings.name = name;
Command::none()
}
Message::EditPackSource(pack_source) => {
self.profile_edit_settings.pack_source = pack_source;
Command::none()
2024-08-21 23:03:56 +01:00
}
Message::SaveProfile => {
let profile: Result<profiles::Profile, String> =
profiles::Profile::try_from(self.profile_edit_settings.clone());
if let Ok(profile) = profile {
if self.profile_edit_settings.name.trim().len() == 0 {
self.profile_save_error =
format!("Invalid profile name {}", self.profile_edit_settings.name)
.into();
} else {
self.userdata
.add_profile(self.profile_edit_settings.name.trim(), profile);
2024-08-24 03:13:59 +01:00
let save_result = self.userdata.save();
if let Err(err) = save_result {
self.profile_save_error =
format!("Unable to save profile: {err:#?}").into();
} else {
self.current_view = ManagerView::ProfileView {
profile: self.profile_edit_settings.name.trim().into(),
}
}
}
} else if let Err(err) = profile {
self.profile_save_error = err.into();
};
Command::none()
2024-08-21 23:03:56 +01:00
}
Message::DeleteProfile(name) => {
self.userdata.remove_profile(&name);
let save_result = self.userdata.save();
if let Err(err) = save_result {
self.profile_save_error = Some(err.to_string());
} else {
self.current_view = ManagerView::ProfileSelect;
}
Command::none()
}
Message::InstallProfile(name) => {
self.current_install_status = ProfileInstallStatus::Installing;
let profile_name = name.clone();
let profile = self.userdata.get_profile(&name).cloned();
Command::perform(
async move {
if let Some(profile) = profile {
let result = profile.install().await;
if let Err(err) = result {
ProfileInstallStatus::Error(format!("{}", err))
}
else {
ProfileInstallStatus::Success
}
}
else {
ProfileInstallStatus::Error(format!("Profile '{}' doesn't exist", name))
}
},
Message::ProfileInstalled
)
},
Message::ProfileInstalled(result) => {
self.current_install_status = result;
Command::none()
},
2024-08-21 23:03:56 +01:00
}
}
fn view(&self) -> Element<Message> {
let contents = match &self.current_view {
ManagerView::ProfileSelect => self.view_profile_select(),
ManagerView::ProfileView { profile } => self.view_profile_view(&profile),
ManagerView::AddProfile => self.view_profile_edit("", ManagerView::ProfileSelect, true),
2024-08-21 23:03:56 +01:00
ManagerView::EditProfile { profile } => self.view_profile_edit(
&profile,
ManagerView::ProfileView {
profile: profile.clone(),
},
false,
2024-08-21 23:03:56 +01:00
),
};
container(contents)
.width(Length::Fill)
.height(Length::Fill)
.center_x()
.into()
}
fn theme(&self) -> Theme {
self.theme.clone()
}
}
impl ManagerGUI {
fn view_profile_select(&self) -> Element<Message> {
let mut profile_select = column![text("Profile Select"),];
let mut profiles_list: Column<Message> = column!();
let mut profile_names = self.userdata.get_profile_names();
profile_names.sort();
for profile_name in profile_names.iter() {
profiles_list = profiles_list.push(
button(text(profile_name))
.on_press(Message::SwitchView(ManagerView::ProfileView {
profile: profile_name.into(),
}))
.width(Length::Fill),
);
}
profile_select =
profile_select.push(profiles_list.align_items(Alignment::Center).spacing(1));
profile_select = profile_select
.push(button("Add profile").on_press(Message::SwitchView(ManagerView::AddProfile)));
scrollable(
profile_select
.spacing(10)
.align_items(Alignment::Center)
.padding(10),
)
2024-08-21 23:03:56 +01:00
.into()
}
fn view_profile_view(&self, profile_name: &str) -> Element<Message> {
let mut profile_view = if let Some(profile) = self.userdata.get_profile(profile_name) {
column![
text(format!("Modpack Profile: {profile_name}"))
.horizontal_alignment(iced::alignment::Horizontal::Center),
row![
"Modpack source",
text_input("Modpack source", &profile.pack_source.to_string()),
]
.spacing(5),
row![
"Mods directory",
text_input("Mods directory", &profile.mods_folder.display().to_string()),
]
.spacing(20),
row!["Mods to download", text(profile.side),].spacing(5),
button("Install").on_press(Message::InstallProfile(profile_name.into())),
row![
button("Back").on_press(Message::SwitchView(ManagerView::ProfileSelect)),
button("Edit profile").on_press(Message::SwitchView(
ManagerView::EditProfile {
profile: profile_name.into()
}
)),
button("Delete profile").on_press(Message::DeleteProfile(profile_name.into()))
]
.spacing(5)
]
} else {
column![
text(format!("Unable to load profile: {profile_name}")),
button("Back").on_press(Message::SwitchView(ManagerView::ProfileSelect)),
]
};
if let Some(err) = &self.userdata_load_error {
profile_view = profile_view.push(text(err));
}
match &self.current_install_status {
ProfileInstallStatus::NotStarted => {},
ProfileInstallStatus::Installing => {
profile_view = profile_view.push(text("Installing..."));
},
ProfileInstallStatus::Success => {
profile_view = profile_view.push(text("Installed"));
},
ProfileInstallStatus::Error(err) => {
profile_view = profile_view.push(text(format!("Failed to install profile: {}", err)));
},
};
profile_view
.align_items(Alignment::Center)
.spacing(10)
.padding(20)
.into()
2024-08-21 23:03:56 +01:00
}
fn view_profile_edit(
&self,
profile_name: &str,
previous_view: ManagerView,
can_edit_name: bool,
2024-08-21 23:03:56 +01:00
) -> Element<Message> {
let current_mods_directory_display = match &self.profile_edit_settings.mods_dir {
Some(mods_dir) => mods_dir.display().to_string(),
None => String::from(""),
};
2024-08-21 23:03:56 +01:00
let mut profile_editor = column![
text("Profile Add/Edit").horizontal_alignment(iced::alignment::Horizontal::Center),
2024-08-21 23:03:56 +01:00
row![
"Profile name",
if can_edit_name {
text_input("Enter your profile name", &self.profile_edit_settings.name)
.on_input(Message::EditProfileName)
} else {
text_input("Profile name", &self.profile_edit_settings.name)
}
2024-08-21 23:03:56 +01:00
]
.spacing(5),
row![
"Modpack source",
text_input(
"Enter a modpack source. E.g git+https://github.com/WarrenHood/SomeModPack",
&self.profile_edit_settings.pack_source
)
.on_input(Message::EditPackSource)
]
.spacing(5),
row![
"Mods directory",
text_input(
"Browse for your MC instance's mods directory",
&current_mods_directory_display
),
2024-08-21 23:03:56 +01:00
button("Browse").on_press(Message::BrowseModsDir)
]
.spacing(5),
row![
button("Back").on_press(Message::SwitchView(previous_view)),
button("Save").on_press(Message::SaveProfile)
]
.spacing(10)
]
.align_items(Alignment::Center)
2024-08-21 23:03:56 +01:00
.spacing(10)
.padding(20);
if let Some(save_error) = &self.profile_save_error {
profile_editor =
profile_editor.extend([row!["Save error", text(save_error)].spacing(10).into()]);
};
profile_editor.into()
}
}