forked from akanyan/STARTLINER
		
	Compare commits
	
		
			4 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| b75cc8f240 | |||
| 407b34a884 | |||
| 890d26e883 | |||
| 2aff5834b9 | 
							
								
								
									
										10
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @ -1,3 +1,13 @@ | ||||
| ## 0.13.0 | ||||
|  | ||||
| - Added profile imports/exports | ||||
| - Fixed error when trying to open an empty Ongeki profile | ||||
| - Switched the default color scheme from invisible to purple | ||||
|  | ||||
| ## 0.12.1 | ||||
|  | ||||
| - Chunithm: fixed crash when using mempatcher | ||||
|  | ||||
| ## 0.12.0 | ||||
|  | ||||
| - Ongeki: cache and mu3.ini config are now split per-profile (requires mu3-mods 3.7+) | ||||
|  | ||||
| @ -1,3 +1,5 @@ | ||||
| Looking for maimai DX players willing to help develop/test maimai DX support | ||||
|  | ||||
| # STARTLINER | ||||
|  | ||||
| This is a program that seeks to streamline game data configuration, currently supporting O.N.G.E.K.I. and CHUNITHM. | ||||
|  | ||||
| @ -1,7 +1,8 @@ | ||||
| use std::hash::{DefaultHasher, Hash, Hasher}; | ||||
| use std::path::Path; | ||||
| use std::time::SystemTime; | ||||
| use crate::model::config::GlobalConfig; | ||||
| use crate::model::patch::PatchFileVec; | ||||
| use crate::model::patch::{PatchFileVec, PatchList}; | ||||
| use crate::pkg::{Feature, Status}; | ||||
| use crate::profiles::types::Profile; | ||||
| use crate::{model::misc::Game, pkg::PkgKey}; | ||||
| @ -165,4 +166,22 @@ impl AppData { | ||||
|             panic!("unable to initialize the logger? {:?}", e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn patches_enabled(&self, game_target: impl AsRef<Path>, amd_target: impl AsRef<Path>) -> Result<Vec<&PatchList>> { | ||||
|         let ch1 = sha256::try_digest(game_target.as_ref())?; | ||||
|         let ch2 = sha256::try_digest(amd_target.as_ref())?; | ||||
|  | ||||
|         let mut res = Vec::new(); | ||||
|         for pfile in &self.patch_vec.0 { | ||||
|             for plist in &pfile.0 { | ||||
|                 let this_hash = plist.sha256.to_ascii_lowercase(); | ||||
|                 log::debug!("checking {}", this_hash); | ||||
|                 if this_hash == ch1 || this_hash == ch2 { | ||||
|                     log::debug!("enabling {this_hash}"); | ||||
|                     res.push(plist); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         Ok(res) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -69,9 +69,15 @@ pub async fn startline(app: AppHandle, refresh: bool) -> Result<(), String> { | ||||
|     } | ||||
|     if let Some(p) = &appd.profile { | ||||
|         log::debug!("{}", hash); | ||||
|  | ||||
|         let patches_enabled = appd.patches_enabled( | ||||
|             &p.data.sgt.target, | ||||
|             &p.data.sgt.target.parent().unwrap().join("amdaemon.exe") | ||||
|         ).map_err(|e| e.to_string())?; | ||||
|  | ||||
|         let info = p.prepare_display() | ||||
|             .map_err(|e| e.to_string())?; | ||||
|         let lineup_res = p.line_up(hash, refresh, &appd.patch_vec).await | ||||
|         let lineup_res = p.line_up(hash, refresh, patches_enabled).await | ||||
|             .map_err(|e| e.to_string()); | ||||
|  | ||||
|         #[cfg(target_os = "windows")] | ||||
| @ -404,6 +410,33 @@ pub async fn create_shortcut(app: AppHandle, profile_meta: ProfileMeta) -> Resul | ||||
|     util::create_shortcut(app, &profile_meta).map_err(|e| e.to_string()) | ||||
| } | ||||
|  | ||||
| #[tauri::command] | ||||
| pub async fn export_profile(state: State<'_, Mutex<AppData>>, export_keychip: bool, files: Vec<String>) -> Result<(), String> { | ||||
|     log::debug!("invoke: export_profile({:?}, {:?} files)", export_keychip, files.len()); | ||||
|  | ||||
|     let appd = state.lock().await; | ||||
|     match &appd.profile { | ||||
|         Some(p) => { | ||||
|             p.export(export_keychip, files) | ||||
|                 .map_err(|e| e.to_string())?; | ||||
|         } | ||||
|         None => { | ||||
|             let err = "export_profile: no profile".to_owned(); | ||||
|             log::error!("{}", err); | ||||
|             return Err(err); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     Ok(()) | ||||
| } | ||||
|  | ||||
| #[tauri::command] | ||||
| pub async fn import_profile(state: State<'_, Mutex<AppData>>, path: PathBuf) -> Result<(), String> { | ||||
|     log::debug!("invoke: import_profile({:?})", path); | ||||
|  | ||||
|     Profile::import(path).map_err(|e| e.to_string()) | ||||
| } | ||||
|  | ||||
| #[tauri::command] | ||||
| pub async fn list_platform_capabilities() -> Result<Vec<String>, ()> { | ||||
|     log::debug!("invoke: list_platform_capabilities"); | ||||
|  | ||||
| @ -205,6 +205,8 @@ pub async fn run(_args: Vec<String>) { | ||||
|             cmd::save_current_profile, | ||||
|             cmd::load_segatools_ini, | ||||
|             cmd::create_shortcut, | ||||
|             cmd::export_profile, | ||||
|             cmd::import_profile, | ||||
|  | ||||
|             cmd::get_global_config, | ||||
|             cmd::set_global_config, | ||||
|  | ||||
| @ -1,18 +1,17 @@ | ||||
| use std::path::Path; | ||||
| use anyhow::Result; | ||||
| use crate::model::patch::{Patch, PatchData, PatchFileVec, PatchSelection, PatchSelectionData}; | ||||
| use crate::model::patch::{Patch, PatchData, PatchList, PatchSelection, PatchSelectionData}; | ||||
|  | ||||
| impl PatchSelection { | ||||
|     pub async fn render_to_file( | ||||
|         &self, | ||||
|         filename: &str, | ||||
|         patches: &PatchFileVec, | ||||
|         patch_lists: &Vec<&PatchList>, | ||||
|         path: impl AsRef<Path> | ||||
|     ) -> Result<()> { | ||||
|         let mut res = "".to_owned(); | ||||
|  | ||||
|         for file in &patches.0 { | ||||
|             for list in &file.0 { | ||||
|         for list in patch_lists { | ||||
|             if list.filename != filename { | ||||
|                 continue; | ||||
|             } | ||||
| @ -22,7 +21,6 @@ impl PatchSelection { | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         } | ||||
|  | ||||
|         tokio::fs::write(path, res).await?; | ||||
|  | ||||
|  | ||||
| @ -30,20 +30,21 @@ impl PatchFileVec { | ||||
|     pub fn find_patches(&self, target: impl AsRef<Path>) -> Result<Vec<Patch>> { | ||||
|         let checksum = try_digest(target.as_ref())?; | ||||
|  | ||||
|         let mut res = Vec::new(); | ||||
|         let mut res_patches = Vec::new(); | ||||
|         for pfile in &self.0 { | ||||
|             for plist in &pfile.0 { | ||||
|                 log::debug!("checking {}", plist.sha256.to_ascii_lowercase()); | ||||
|                 if plist.sha256.to_ascii_lowercase() == checksum { | ||||
|                 let this_hash = plist.sha256.to_ascii_lowercase(); | ||||
|                 log::debug!("checking {}", this_hash); | ||||
|                 if this_hash == checksum { | ||||
|                     let mut cloned = plist.clone().patches; | ||||
|                     res.append(&mut cloned); | ||||
|                     res_patches.append(&mut cloned); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if res.len() == 0 { | ||||
|         if res_patches.len() == 0 { | ||||
|             log::warn!("no matching patchset for {:?} ({})", target.as_ref(), checksum); | ||||
|         } | ||||
|         Ok(res) | ||||
|         Ok(res_patches) | ||||
|     } | ||||
| } | ||||
| @ -1,6 +1,6 @@ | ||||
| pub use types::{Profile, ProfileData, ProfileMeta, ProfilePaths, StartPayload}; | ||||
| use std::{collections::{BTreeMap, BTreeSet}, path::{Path, PathBuf}}; | ||||
| use crate::{model::{misc::Game, patch::{PatchFileVec, PatchSelection}, profile::{Aime, ChunithmKeyboard, IOSelection, Keyboard, Mu3Ini, OngekiKeyboard, ProfileModule}}, modules::{display_windows::DisplayInfo, package::prepare_packages}, pkg::PkgKey, pkg_store::PackageStore, util}; | ||||
| use crate::{model::{misc::Game, patch::{PatchList, PatchSelection}, profile::{Aime, ChunithmKeyboard, IOSelection, Keyboard, Mu3Ini, OngekiKeyboard, ProfileModule}}, modules::{display_windows::DisplayInfo, package::prepare_packages}, pkg::PkgKey, pkg_store::PackageStore, util}; | ||||
| use tauri::Emitter; | ||||
| use std::process::Stdio; | ||||
| use crate::model::profile::BepInEx; | ||||
| @ -10,6 +10,7 @@ use std::fs::File; | ||||
| use tokio::process::Command; | ||||
| use tokio::task::JoinSet; | ||||
|  | ||||
| pub mod template; | ||||
| pub mod types; | ||||
|  | ||||
| impl Profile { | ||||
| @ -43,12 +44,6 @@ impl Profile { | ||||
|         std::fs::create_dir_all(p.config_dir())?; | ||||
|         std::fs::create_dir_all(p.data_dir())?; | ||||
|  | ||||
|         if meta.game == Game::Ongeki { | ||||
|             if let Err(e) = Self::load_existing_mu3_ini(&p.data, &p.meta) { | ||||
|                 log::error!("unable to load existing mu3.ini: {e}"); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         match meta.game { | ||||
|             Game::Ongeki => std::fs::write(p.config_dir().join("segatools-base.ini"), include_bytes!("../../static/segatools-ongeki.ini"))?, | ||||
|             Game::Chunithm => std::fs::write(p.config_dir().join("segatools-base.ini"), include_bytes!("../../static/segatools-chunithm.ini"))?, | ||||
| @ -186,7 +181,7 @@ impl Profile { | ||||
|  | ||||
|         Ok(info) | ||||
|     } | ||||
|     pub async fn line_up(&self, pkg_hash: String, refresh: bool, patch_files: &PatchFileVec) -> Result<()> { | ||||
|     pub async fn line_up(&self, pkg_hash: String, refresh: bool, patchlists_enabled: Vec<&PatchList>) -> Result<()> { | ||||
|         if !self.data_dir().exists() { | ||||
|             tokio::fs::create_dir(self.data_dir()).await?; | ||||
|         } | ||||
| @ -226,8 +221,8 @@ impl Profile { | ||||
|  | ||||
|         if let Some(patches) = &self.data.patches { | ||||
|             futures::try_join!( | ||||
|                 patches.render_to_file("amdaemon.exe", patch_files, self.data_dir().join("patch-amd.mph")), | ||||
|                 patches.render_to_file("chusanApp.exe", patch_files, self.data_dir().join("patch-game.mph")) | ||||
|                 patches.render_to_file("amdaemon.exe", &patchlists_enabled, self.data_dir().join("patch-amd.mph")), | ||||
|                 patches.render_to_file("chusanApp.exe", &patchlists_enabled, self.data_dir().join("patch-game.mph")) | ||||
|             )?; | ||||
|         } | ||||
|  | ||||
| @ -431,13 +426,15 @@ impl Profile { | ||||
|     } | ||||
|  | ||||
|     fn load_existing_mu3_ini(data: &ProfileData, meta: &ProfileMeta) -> Result<()> { | ||||
|         let mu3_ini_target_path = data.sgt.target.parent().ok_or_else(|| anyhow!("invalid target directory"))?.join("mu3.ini"); | ||||
|         if let Some(parent) =  data.sgt.target.parent() { | ||||
|             let mu3_ini_target_path = parent.join("mu3.ini"); | ||||
|             let mu3_ini_profile_path = util::profile_config_dir(meta.game, &meta.name).join("mu3.ini"); | ||||
|             log::debug!("mu3.ini paths: {:?} {:?}", mu3_ini_target_path, mu3_ini_profile_path); | ||||
|             if mu3_ini_target_path.exists() && !mu3_ini_profile_path.exists() { | ||||
|                 std::fs::copy(&mu3_ini_target_path, &mu3_ini_profile_path)?; | ||||
|                 log::info!("copied mu3.ini from {:?}", &mu3_ini_target_path); | ||||
|             } | ||||
|         } | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
|  | ||||
							
								
								
									
										90
									
								
								rust/src/profiles/template.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								rust/src/profiles/template.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,90 @@ | ||||
| use std::{fs::File, io::{Read, Write}, path::PathBuf}; | ||||
| use zip::{write::FileOptions, ZipArchive, ZipWriter}; | ||||
| use crate::util; | ||||
| use super::{Profile, ProfilePaths}; | ||||
|  | ||||
| impl Profile { | ||||
|     fn find_template_json(archive: &mut ZipArchive<File>) -> anyhow::Result<String> { | ||||
|         if let Ok(mut file) = archive.by_name("template.json") { | ||||
|             let mut contents = Vec::new(); | ||||
|             file.read_to_end(&mut contents)?; | ||||
|             Ok(String::from_utf8(contents)?) | ||||
|         } else { | ||||
|             anyhow::bail!("invalid template: no template.json found") | ||||
|         } | ||||
|     } | ||||
|     pub fn import(path: PathBuf) -> anyhow::Result<()> { | ||||
|         let file = File::open(path)?; | ||||
|         let mut archive = ZipArchive::new(file)?; | ||||
|  | ||||
|         match Self::find_template_json(&mut archive) { | ||||
|             Ok(raw_p) => { | ||||
|                 let p = serde_json::from_str::<Profile>(&raw_p)?; | ||||
|                 let dir = util::config_dir().join(format!("profile-{}-{}", &p.meta.game, &p.meta.name)); | ||||
|                 if dir.exists() { | ||||
|                     anyhow::bail!("profile {} already exists", &p.meta.name); | ||||
|                 } | ||||
|                 std::fs::create_dir(&dir)?; | ||||
|                 archive.extract(&dir)?; | ||||
|                 std::fs::remove_file(dir.join("template.json"))?; | ||||
|                 std::fs::write(dir.join("profile.json"), serde_json::to_string_pretty(&p.data)?)?; | ||||
|             } | ||||
|             Err(e) => { | ||||
|                 return Err(e); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         Ok(()) | ||||
|     } | ||||
|     pub fn export(&self, export_keychip: bool, extra_files: Vec<String>) -> anyhow::Result<()> { | ||||
|         let mut prf = self.clone(); | ||||
|  | ||||
|         let dir = util::config_dir().join("exports"); | ||||
|  | ||||
|         if !dir.exists() { | ||||
|             std::fs::create_dir(&dir)?; | ||||
|         } | ||||
|  | ||||
|         let path = dir.join(format!("{}-{}-template.zip", &self.meta.game, &self.meta.name)); | ||||
|  | ||||
|         { | ||||
|             let sgt = &mut prf.data.sgt; | ||||
|             sgt.target = PathBuf::new(); | ||||
|             if sgt.amfs.is_absolute() { | ||||
|                 sgt.amfs = PathBuf::new(); | ||||
|             } | ||||
|             if sgt.option.is_absolute() { | ||||
|                 sgt.option = PathBuf::new(); | ||||
|             } | ||||
|             if sgt.appdata.is_absolute() { | ||||
|                 sgt.appdata = PathBuf::new(); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         { | ||||
|             let network = &mut prf.data.network; | ||||
|             if network.local_path.is_absolute() { | ||||
|                 network.local_path = PathBuf::new(); | ||||
|             } | ||||
|             if !export_keychip { | ||||
|                 network.keychip = String::new(); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         let file = File::create(&path)?; | ||||
|         let mut zip = ZipWriter::new(file); | ||||
|         let options: FileOptions<'_, ()> = FileOptions::default(); | ||||
|         zip.start_file("template.json", options)?; | ||||
|         zip.write_all(&serde_json::to_string_pretty(&prf)?.as_bytes())?; | ||||
|  | ||||
|         for file in extra_files { | ||||
|             log::debug!("extra file: {file}"); | ||||
|             zip.start_file(&file, options)?; | ||||
|             zip.write_all(&std::fs::read(self.config_dir().join(file))?)?; | ||||
|         } | ||||
|  | ||||
|         zip.finish()?; | ||||
|  | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
| @ -1,7 +1,7 @@ | ||||
| { | ||||
|     "$schema": "https://schema.tauri.app/config/2", | ||||
|     "productName": "STARTLINER", | ||||
|     "version": "0.12.0", | ||||
|     "version": "0.13.0", | ||||
|     "identifier": "zip.patafour.startliner", | ||||
|     "build": { | ||||
|         "beforeDevCommand": "bun run dev", | ||||
|  | ||||
| @ -1,29 +1,154 @@ | ||||
| <script setup lang="ts"> | ||||
| import { Ref, ref } from 'vue'; | ||||
| import Button from 'primevue/button'; | ||||
| import Dialog from 'primevue/dialog'; | ||||
| import ToggleSwitch from 'primevue/toggleswitch'; | ||||
| import * as path from '@tauri-apps/api/path'; | ||||
| import { open } from '@tauri-apps/plugin-dialog'; | ||||
| import ProfileListEntry from './ProfileListEntry.vue'; | ||||
| import { usePrfStore } from '../stores'; | ||||
| import { invoke } from '../invoke'; | ||||
| import { useClientStore, useGeneralStore, usePrfStore } from '../stores'; | ||||
|  | ||||
| const prf = usePrfStore(); | ||||
| const client = useClientStore(); | ||||
| const general = useGeneralStore(); | ||||
|  | ||||
| const exportVisible = ref(false); | ||||
| const exportKeychip = ref(false); | ||||
| const files = new Set<string>(); | ||||
|  | ||||
| const exportTemplate = async () => { | ||||
|     const fl = [...files.values()]; | ||||
|     exportVisible.value = false; | ||||
|     await invoke('export_profile', { | ||||
|         exportKeychip: exportKeychip.value, | ||||
|         files: fl, | ||||
|     }); | ||||
|     await invoke('open_file', { | ||||
|         path: await path.join(general.configDir, 'exports'), | ||||
|     }); | ||||
| }; | ||||
|  | ||||
| const fileList = { | ||||
|     ongeki: ['aime.txt', 'inohara.cfg', 'mu3.ini', 'segatools-base.ini'], | ||||
|     chunithm: ['aime.txt', 'saekawa.toml', 'segatools-base.ini'], | ||||
| }; | ||||
|  | ||||
| const fileListCurrent: Ref<string[]> = ref([]); | ||||
|  | ||||
| const recalcFileList = async () => { | ||||
|     const res: string[] = []; | ||||
|     files.clear(); | ||||
|     for (const idx in fileList[prf.current!.meta.game]) { | ||||
|         const f = fileList[prf.current!.meta.game][idx]; | ||||
|         const p = await path.join(await prf.configDir, f); | ||||
|         if (await invoke('file_exists', { path: p })) { | ||||
|             res.push(f); | ||||
|             files.add(f); | ||||
|         } | ||||
|     } | ||||
|     fileListCurrent.value = res; | ||||
| }; | ||||
|  | ||||
| const openExportDialog = async () => { | ||||
|     await recalcFileList(); | ||||
|     exportVisible.value = true; | ||||
| }; | ||||
|  | ||||
| const importPick = async () => { | ||||
|     const res = await open({ | ||||
|         multiple: false, | ||||
|         directory: false, | ||||
|         filters: [ | ||||
|             { | ||||
|                 name: 'STARTLINER template', | ||||
|                 extensions: ['zip'], | ||||
|             }, | ||||
|         ], | ||||
|     }); | ||||
|     if (res != null) { | ||||
|         await invoke('import_profile', { path: res }); | ||||
|         await prf.reloadList(); | ||||
|     } | ||||
| }; | ||||
| </script> | ||||
|  | ||||
| <template> | ||||
|     <Dialog | ||||
|         modal | ||||
|         :visible="exportVisible" | ||||
|         :closable="false /*this shit doesn't work */" | ||||
|         :header="`Export ${prf.current?.meta.name}`" | ||||
|         :style="{ width: '300px', scale: client.scaleValue }" | ||||
|     > | ||||
|         <div class="flex flex-col gap-4"> | ||||
|             <div class="flex flex-row"> | ||||
|                 <div class="grow">Export keychip</div> | ||||
|                 <ToggleSwitch v-model="exportKeychip" /> | ||||
|             </div> | ||||
|             <div class="flex flex-row" v-for="f in fileListCurrent"> | ||||
|                 <div class="grow">Export {{ f }}</div> | ||||
|                 <ToggleSwitch | ||||
|                     :model-value="true" | ||||
|                     @update:model-value=" | ||||
|                         (v) => { | ||||
|                             if (v === true) { | ||||
|                                 files.add(f); | ||||
|                             } else { | ||||
|                                 files.delete(f); | ||||
|                             } | ||||
|                         } | ||||
|                     " | ||||
|                 /> | ||||
|             </div> | ||||
|             <div style="width: 100%; text-align: center"> | ||||
|                 <Button | ||||
|                     class="m-auto mr-3" | ||||
|                     style="width: 80px" | ||||
|                     label="OK" | ||||
|                     @click="() => exportTemplate()" | ||||
|                 /> | ||||
|                 <Button | ||||
|                     class="m-auto" | ||||
|                     style="width: 80px" | ||||
|                     label="Cancel" | ||||
|                     @click="() => (exportVisible = false)" | ||||
|                 /> | ||||
|             </div> | ||||
|         </div> | ||||
|     </Dialog> | ||||
|     <div v-if="prf.list.length === 0"> | ||||
|         Welcome to STARTLINER! Start by creating a profile. | ||||
|     </div> | ||||
|     <div class="mt-4 flex flex-row flex-wrap align-middle gap-4"> | ||||
|         <Button | ||||
|             label="O.N.G.E.K.I. profile" | ||||
|             icon="pi pi-plus" | ||||
|             icon="pi pi-file-plus" | ||||
|             class="ongeki-button profile-button" | ||||
|             @click="() => prf.create('ongeki')" | ||||
|         /> | ||||
|         <Button | ||||
|             label="CHUNITHM profile" | ||||
|             icon="pi pi-plus" | ||||
|             icon="pi pi-file-plus" | ||||
|             class="chunithm-button profile-button" | ||||
|             @click="() => prf.create('chunithm')" | ||||
|         /> | ||||
|     </div> | ||||
|     <div class="mt-4 flex flex-row flex-wrap align-middle gap-4"> | ||||
|         <Button | ||||
|             label="Import template" | ||||
|             icon="pi pi-file-import" | ||||
|             class="import-button profile-button" | ||||
|             @click="() => importPick()" | ||||
|         /> | ||||
|         <Button | ||||
|             :disabled="prf.current === null" | ||||
|             label="Export template" | ||||
|             icon="pi pi-file-export" | ||||
|             class="profile-button" | ||||
|             @click="() => openExportDialog()" | ||||
|         /> | ||||
|     </div> | ||||
|     <div class="mt-12 flex flex-col flex-wrap align-middle gap-4"> | ||||
|         <div v-for="p in prf.list"> | ||||
|             <ProfileListEntry :p="p" /> | ||||
| @ -57,4 +182,14 @@ const prf = usePrfStore(); | ||||
|     background-color: var(--p-yellow-300) !important; | ||||
|     border-color: var(--p-yellow-300) !important; | ||||
| } | ||||
|  | ||||
| .import-button { | ||||
|     background-color: var(--p-purple-400) !important; | ||||
|     border-color: var(--p-purple-400) !important; | ||||
| } | ||||
| .import-button:hover, | ||||
| .import-button:active { | ||||
|     background-color: var(--p-purple-300) !important; | ||||
|     border-color: var(--p-purple-300) !important; | ||||
| } | ||||
| </style> | ||||
|  | ||||
| @ -3,11 +3,7 @@ import { Feature, Game, Package } from './types'; | ||||
|  | ||||
| export const changePrimaryColor = (game: Game | null) => { | ||||
|     const color = | ||||
|         game === 'ongeki' | ||||
|             ? 'pink' | ||||
|             : game === 'chunithm' | ||||
|               ? 'yellow' | ||||
|               : 'bluegray'; | ||||
|         game === 'ongeki' ? 'pink' : game === 'chunithm' ? 'yellow' : 'purple'; | ||||
|  | ||||
|     updatePrimaryPalette({ | ||||
|         50: `{${color}.50}`, | ||||
|  | ||||
		Reference in New Issue
	
	Block a user
	