1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465 |
- using UnityEngine;
- using System.Linq;
- using System;
- using System.Collections.Generic;
- using UnityEngine.UI;
- using HMUI;
- using VRUI;
- using SongBrowser.DataAccess;
- using TMPro;
- using Logger = SongBrowser.Logging.Logger;
- using SongBrowser.DataAccess.BeatSaverApi;
- using System.Collections;
- using SongCore.Utilities;
- using SongBrowser.Internals;
- using CustomUI.BeatSaber;
- using SongDataCore.ScoreSaber;
- namespace SongBrowser.UI
- {
- public enum UIState
- {
- Disabled,
- Main,
- SortBy,
- FilterBy
- }
- /// <summary>
- /// Hijack the flow coordinator. Have access to all StandardLevel easily.
- /// </summary>
- public class SongBrowserUI : MonoBehaviour
- {
- // Logging
- public const String Name = "SongBrowserUI";
- private const float SEGMENT_PERCENT = 0.1f;
- private const int LIST_ITEMS_VISIBLE_AT_ONCE = 6;
- // BeatSaber Internal UI structures
- DataAccess.BeatSaberUIController _beatUi;
- // New UI Elements
- private List<SongSortButton> _sortButtonGroup;
- private List<SongFilterButton> _filterButtonGroup;
- private Button _sortByButton;
- private Button _sortByDisplay;
- private Button _filterByButton;
- private Button _filterByDisplay;
- private Button _randomButton;
- private Button _clearSortFilterButton;
- private Button _addFavoriteButton;
- private SimpleDialogPromptViewController _deleteDialog;
- private Button _deleteButton;
- private Button _pageUpFastButton;
- private Button _pageDownFastButton;
- private SearchKeyboardViewController _searchViewController;
- private PlaylistFlowCoordinator _playListFlowCoordinator;
- private RectTransform _ppStatButton;
- private RectTransform _starStatButton;
- private RectTransform _njsStatButton;
-
- private Sprite _currentAddFavoriteButtonSprite;
- // Model
- private SongBrowserModel _model;
- public SongBrowserModel Model
- {
- get
- {
- return _model;
- }
- }
- // UI Created
- private bool _uiCreated = false;
- private UIState _currentUiState = UIState.Disabled;
- /// <summary>
- /// Constructor
- /// </summary>
- public SongBrowserUI() : base()
- {
- if (_model == null)
- {
- _model = new SongBrowserModel();
- }
- _model.Init();
- }
- /// <summary>
- /// Builds the UI for this plugin.
- /// </summary>
- public void CreateUI(MainMenuViewController.MenuButton mode)
- {
- Logger.Trace("CreateUI()");
- // Determine the flow controller to use
- FlowCoordinator flowCoordinator = null;
- if (mode == MainMenuViewController.MenuButton.SoloFreePlay)
- {
- Logger.Debug("Entering SOLO mode...");
- flowCoordinator = Resources.FindObjectsOfTypeAll<SoloFreePlayFlowCoordinator>().First();
- }
- else if (mode == MainMenuViewController.MenuButton.Party)
- {
- Logger.Debug("Entering PARTY mode...");
- flowCoordinator = Resources.FindObjectsOfTypeAll<PartyFreePlayFlowCoordinator>().First();
- }
- else
- {
- Logger.Debug("Entering SOLO CAMPAIGN mode...");
- flowCoordinator = Resources.FindObjectsOfTypeAll<CampaignFlowCoordinator>().First();
- return;
- }
- _beatUi = new DataAccess.BeatSaberUIController(flowCoordinator);
- // returning to the menu and switching modes could trigger this.
- if (_uiCreated)
- {
- return;
- }
- try
- {
- if (_playListFlowCoordinator == null)
- {
- _playListFlowCoordinator = UIBuilder.CreateFlowCoordinator<PlaylistFlowCoordinator>("PlaylistFlowCoordinator");
- _playListFlowCoordinator.didFinishEvent += HandleDidSelectPlaylist;
- }
- // delete dialog
- this._deleteDialog = UnityEngine.Object.Instantiate<SimpleDialogPromptViewController>(_beatUi.SimpleDialogPromptViewControllerPrefab);
- this._deleteDialog.name = "DeleteDialogPromptViewController";
- this._deleteDialog.gameObject.SetActive(false);
- // create song browser main ui
- CreateOuterUi();
- CreateSortButtons();
- CreateFilterButtons();
- CreateAddFavoritesButton();
- CreateDeleteButton();
- CreateFastPageButtons();
- this.InstallHandlers();
- this.ModifySongStatsPanel();
- this.ResizeSongUI();
- RefreshSortButtonUI();
- _uiCreated = true;
- Logger.Debug("Done Creating UI...");
- }
- catch (Exception e)
- {
- Logger.Exception("Exception during CreateUI: ", e);
- }
- }
- /// <summary>
- /// Create the outer ui.
- /// </summary>
- private void CreateOuterUi()
- {
- Logger.Debug("Creating outer UI...");
- float clearButtonX = -32.5f;
- float clearButtonY = 34.5f;
- float buttonY = 37f;
- float buttonHeight = 5.0f;
- float sortByButtonX = -22.5f + buttonHeight;
- float outerButtonFontSize = 3.0f;
- float displayButtonFontSize = 2.5f;
- float outerButtonWidth = 24.0f;
- float randomButtonWidth = 8.0f;
- // clear button
- _clearSortFilterButton = CreateClearButton(clearButtonX, clearButtonY, buttonHeight, () =>
- {
- if (_currentUiState == UIState.FilterBy || _currentUiState == UIState.SortBy)
- {
- RefreshOuterUIState(UIState.Main);
- }
- else
- {
- OnClearButtonClickEvent();
- }
- });
- // create SortBy button and its display
- float curX = sortByButtonX;
- _sortByButton = _beatUi.LevelPackLevelsViewController.CreateUIButton("ApplyButton", new Vector2(curX, buttonY), new Vector2(outerButtonWidth, buttonHeight), () =>
- {
- RefreshOuterUIState(UIState.SortBy);
- }, "Sort By");
- _sortByButton.SetButtonTextSize(outerButtonFontSize);
- _sortByButton.ToggleWordWrapping(false);
- curX += outerButtonWidth;
- _sortByDisplay = _beatUi.LevelPackLevelsViewController.CreateUIButton("ApplyButton", new Vector2(curX, buttonY), new Vector2(outerButtonWidth, buttonHeight), () =>
- {
- OnSortButtonClickEvent(_model.Settings.sortMode);
- }, "");
- _sortByDisplay.SetButtonTextSize(displayButtonFontSize);
- _sortByDisplay.ToggleWordWrapping(false);
- curX += outerButtonWidth;
- // create FilterBy button and its display
- _filterByButton = _beatUi.LevelPackLevelsViewController.CreateUIButton("ApplyButton", new Vector2(curX, buttonY), new Vector2(outerButtonWidth, buttonHeight), () =>
- {
- RefreshOuterUIState(UIState.FilterBy);
- }, "Filter By");
- _filterByButton.SetButtonTextSize(outerButtonFontSize);
- _filterByButton.ToggleWordWrapping(false);
- curX += outerButtonWidth;
- _filterByDisplay = _beatUi.LevelPackLevelsViewController.CreateUIButton("ApplyButton", new Vector2(curX, buttonY), new Vector2(outerButtonWidth, buttonHeight), () =>
- {
- _model.Settings.filterMode = SongFilterMode.None;
- SongCore.Loader.Instance.RefreshLevelPacks();
- RefreshSongUI();
- }, "");
- _filterByDisplay.SetButtonTextSize(displayButtonFontSize);
- _filterByDisplay.ToggleWordWrapping(false);
- // random button
- _randomButton = _beatUi.LevelPackLevelsViewController.CreateUIButton("HowToPlayButton", new Vector2(curX + (outerButtonWidth / 2.0f) + (randomButtonWidth / 2.0f), clearButtonY), new Vector2(randomButtonWidth, buttonHeight), () =>
- {
- OnSortButtonClickEvent(SongSortMode.Random);
- }, "",
- Base64Sprites.RandomIcon);
- _randomButton.GetComponentsInChildren<HorizontalLayoutGroup>().First(btn => btn.name == "Content").padding = new RectOffset(0, 0, 0, 0);
- var textRect = _randomButton.GetComponentsInChildren<RectTransform>(true).FirstOrDefault(c => c.name == "Text");
- if (textRect != null)
- {
- UnityEngine.Object.Destroy(textRect.gameObject);
- }
- UIBuilder.SetButtonBorderActive(_randomButton, false);
- }
- /// <summary>
- /// Create the back button
- /// </summary>
- /// <returns></returns>
- private Button CreateClearButton(float x, float y, float h, UnityEngine.Events.UnityAction callback)
- {
- Button b = _beatUi.LevelPackLevelsViewController.CreateUIButton("HowToPlayButton", new Vector2(x, y), new Vector2(h, h), callback, "", Base64Sprites.XIcon);
- b.GetComponentsInChildren<HorizontalLayoutGroup>().First(btn => btn.name == "Content").padding = new RectOffset(1, 1, 0, 0);
- RectTransform textRect = b.GetComponentsInChildren<RectTransform>(true).FirstOrDefault(c => c.name == "Text");
- if (textRect != null)
- {
- UnityEngine.Object.Destroy(textRect.gameObject);
- }
- UIBuilder.SetButtonBorderActive(b, false);
- return b;
- }
- /// <summary>
- /// Create the sort button ribbon
- /// </summary>
- private void CreateSortButtons()
- {
- Logger.Debug("Create sort buttons...");
- float sortButtonFontSize = 2.15f;
- float sortButtonX = -23.0f;
- float sortButtonWidth = 12.0f;
- float buttonSpacing = 0.25f;
- float buttonY = 37f;
- float buttonHeight = 5.0f;
- string[] sortButtonNames = new string[]
- {
- "Title", "Author", "Newest", "YourPlays", "PP", "Stars", "UpVotes", "PlayCount", "Rating", "Heat"
- };
- SongSortMode[] sortModes = new SongSortMode[]
- {
- SongSortMode.Default, SongSortMode.Author, SongSortMode.Newest, SongSortMode.YourPlayCount, SongSortMode.PP, SongSortMode.Stars, SongSortMode.UpVotes, SongSortMode.PlayCount, SongSortMode.Rating, SongSortMode.Heat
- };
- _sortButtonGroup = new List<SongSortButton>();
- for (int i = 0; i < sortButtonNames.Length; i++)
- {
- float curButtonX = sortButtonX + (sortButtonWidth * i) + (buttonSpacing * i);
- SongSortButton sortButton = new SongSortButton();
- sortButton.SortMode = sortModes[i];
- sortButton.Button = _beatUi.LevelPackLevelsViewController.CreateUIButton("ApplyButton",
- new Vector2(curButtonX, buttonY), new Vector2(sortButtonWidth, buttonHeight),
- () =>
- {
- OnSortButtonClickEvent(sortButton.SortMode);
- RefreshOuterUIState(UIState.Main);
- },
- sortButtonNames[i]);
- sortButton.Button.SetButtonTextSize(sortButtonFontSize);
- sortButton.Button.GetComponentsInChildren<HorizontalLayoutGroup>().First(btn => btn.name == "Content").padding = new RectOffset(4, 4, 2, 2);
- sortButton.Button.ToggleWordWrapping(false);
- sortButton.Button.name = "Sort" + sortModes[i].ToString() + "Button";
- _sortButtonGroup.Add(sortButton);
- }
- }
- /// <summary>
- /// Create the filter by buttons
- /// </summary>
- private void CreateFilterButtons()
- {
- Logger.Debug("Creating filter buttons...");
- float filterButtonFontSize = 2.25f;
- float filterButtonX = -23.0f;
- float filterButtonWidth = 12.25f;
- float buttonSpacing = 0.5f;
- float buttonY = 37f;
- float buttonHeight = 5.0f;
- string[] filterButtonNames = new string[]
- {
- "Favorites", "Playlist", "Search", "Ranked", "Unranked"
- };
- SongFilterMode[] filterModes = new SongFilterMode[]
- {
- SongFilterMode.Favorites, SongFilterMode.Playlist, SongFilterMode.Search, SongFilterMode.Ranked, SongFilterMode.Unranked
- };
- _filterButtonGroup = new List<SongFilterButton>();
- for (int i = 0; i < filterButtonNames.Length; i++)
- {
- float curButtonX = filterButtonX + (filterButtonWidth * i) + (buttonSpacing * i);
- SongFilterButton filterButton = new SongFilterButton();
- filterButton.FilterMode = filterModes[i];
- filterButton.Button = _beatUi.LevelPackLevelsViewController.CreateUIButton("ApplyButton",
- new Vector2(curButtonX, buttonY), new Vector2(filterButtonWidth, buttonHeight),
- () =>
- {
- OnFilterButtonClickEvent(filterButton.FilterMode);
- RefreshOuterUIState(UIState.Main);
- },
- filterButtonNames[i]);
- filterButton.Button.SetButtonTextSize(filterButtonFontSize);
- filterButton.Button.GetComponentsInChildren<HorizontalLayoutGroup>().First(btn => btn.name == "Content").padding = new RectOffset(4, 4, 2, 2);
- filterButton.Button.ToggleWordWrapping(false);
- filterButton.Button.name = "Filter" + filterButtonNames[i] + "Button";
- _filterButtonGroup.Add(filterButton);
- }
- }
- /// <summary>
- /// Create the fast page up and down buttons
- /// </summary>
- private void CreateFastPageButtons()
- {
- Logger.Debug("Creating fast scroll button...");
- _pageUpFastButton = Instantiate(_beatUi.TableViewPageUpButton, _beatUi.LevelPackLevelsTableViewRectTransform, false);
- (_pageUpFastButton.transform as RectTransform).anchorMin = new Vector2(0.5f, 1f);
- (_pageUpFastButton.transform as RectTransform).anchorMax = new Vector2(0.5f, 1f);
- (_pageUpFastButton.transform as RectTransform).anchoredPosition = new Vector2(-26f, 0.25f);
- (_pageUpFastButton.transform as RectTransform).sizeDelta = new Vector2(8f, 6f);
- _pageUpFastButton.GetComponentsInChildren<RectTransform>().First(x => x.name == "BG").sizeDelta = new Vector2(8f, 6f);
- _pageUpFastButton.GetComponentsInChildren<UnityEngine.UI.Image>().First(x => x.name == "Arrow").sprite = Base64Sprites.DoubleArrow;
- _pageUpFastButton.onClick.AddListener(delegate ()
- {
- this.JumpSongList(-1, SEGMENT_PERCENT);
- });
- _pageDownFastButton = Instantiate(_beatUi.TableViewPageDownButton, _beatUi.LevelPackLevelsTableViewRectTransform, false);
- (_pageDownFastButton.transform as RectTransform).anchorMin = new Vector2(0.5f, 0f);
- (_pageDownFastButton.transform as RectTransform).anchorMax = new Vector2(0.5f, 0f);
- (_pageDownFastButton.transform as RectTransform).anchoredPosition = new Vector2(-26f, -1f);
- (_pageDownFastButton.transform as RectTransform).sizeDelta = new Vector2(8f, 6f);
- _pageDownFastButton.GetComponentsInChildren<RectTransform>().First(x => x.name == "BG").sizeDelta = new Vector2(8f, 6f);
- _pageDownFastButton.GetComponentsInChildren<UnityEngine.UI.Image>().First(x => x.name == "Arrow").sprite = Base64Sprites.DoubleArrow;
- _pageDownFastButton.onClick.AddListener(delegate ()
- {
- this.JumpSongList(1, SEGMENT_PERCENT);
- });
- }
- /// <summary>
- /// Create the +/- favorite button in the play button container.
- /// </summary>
- private void CreateAddFavoritesButton()
- {
- // Create add favorite button
- Logger.Debug("Creating Add to favorites button...");
- _addFavoriteButton = UIBuilder.CreateIconButton(_beatUi.PlayButtons, _beatUi.PracticeButton, Base64Sprites.AddToFavoritesIcon);
- _addFavoriteButton.onClick.AddListener(delegate () {
- ToggleSongInPlaylist();
- });
- }
- /// <summary>
- /// Create the delete button in the play button container
- /// </summary>
- private void CreateDeleteButton()
- {
- // Create delete button
- Logger.Debug("Creating delete button...");
- _deleteButton = UIBuilder.CreateIconButton(_beatUi.PlayButtons, _beatUi.PracticeButton, Base64Sprites.DeleteIcon);
- _deleteButton.onClick.AddListener(delegate () {
- HandleDeleteSelectedLevel();
- });
- }
- /// <summary>
- /// Resize the stats panel to fit more stats.
- /// </summary>
- private void ModifySongStatsPanel()
- {
- // modify details view
- Logger.Debug("Resizing Stats Panel...");
- var statsPanel = _beatUi.StandardLevelDetailView.GetPrivateField<LevelParamsPanel>("_levelParamsPanel");
- var statTransforms = statsPanel.GetComponentsInChildren<RectTransform>();
- var valueTexts = statsPanel.GetComponentsInChildren<TextMeshProUGUI>().Where(x => x.name == "ValueText").ToList();
- RectTransform panelRect = (statsPanel.transform as RectTransform);
- panelRect.sizeDelta = new Vector2(panelRect.sizeDelta.x * 1.2f, panelRect.sizeDelta.y * 1.2f);
- for (int i = 0; i < statTransforms.Length; i++)
- {
- var r = statTransforms[i];
- if (r.name == "Separator")
- {
- continue;
- }
- r.sizeDelta = new Vector2(r.sizeDelta.x * 0.75f, r.sizeDelta.y * 0.75f);
- }
- for (int i = 0; i < valueTexts.Count; i++)
- {
- var text = valueTexts[i];
- text.fontSize = 3.25f;
- }
- // inject our components
- _ppStatButton = UnityEngine.Object.Instantiate(statTransforms[1], statsPanel.transform, false);
- UIBuilder.SetStatButtonIcon(_ppStatButton, Base64Sprites.GraphIcon);
- _starStatButton = UnityEngine.Object.Instantiate(statTransforms[1], statsPanel.transform, false);
- UIBuilder.SetStatButtonIcon(_starStatButton, Base64Sprites.StarFullIcon);
- _njsStatButton = UnityEngine.Object.Instantiate(statTransforms[1], statsPanel.transform, false);
- UIBuilder.SetStatButtonIcon(_njsStatButton, Base64Sprites.SpeedIcon);
- // shrink title
- var titleText = _beatUi.LevelDetailViewController.GetComponentsInChildren<TextMeshProUGUI>(true).First(x => x.name == "SongNameText");
- titleText.fontSize = 5.0f;
- }
- /// <summary>
- /// Resize some of the song table elements.
- /// </summary>
- public void ResizeSongUI()
- {
- // Reposition the table view a bit
- _beatUi.LevelPackLevelsTableViewRectTransform.anchoredPosition = new Vector2(0f, -2.5f);
- // Move the page up/down buttons a bit
- TableView tableView = ReflectionUtil.GetPrivateField<TableView>(_beatUi.LevelPackLevelsTableView, "_tableView");
- RectTransform pageUpButton = _beatUi.TableViewPageUpButton.transform as RectTransform;
- RectTransform pageDownButton = _beatUi.TableViewPageDownButton.transform as RectTransform;
- pageUpButton.anchoredPosition = new Vector2(pageUpButton.anchoredPosition.x, pageUpButton.anchoredPosition.y - 1f);
- pageDownButton.anchoredPosition = new Vector2(pageDownButton.anchoredPosition.x, pageDownButton.anchoredPosition.y + 1f);
- // shrink play button container
- RectTransform playContainerRect = _beatUi.StandardLevelDetailView.GetComponentsInChildren<RectTransform>().First(x => x.name == "PlayContainer");
- RectTransform playButtonsRect = playContainerRect.GetComponentsInChildren<RectTransform>().First(x => x.name == "PlayButtons");
- playButtonsRect.localScale = new Vector3(0.825f, 0.825f, 0.825f);
- }
- /// <summary>
- /// Add our handlers into BeatSaber.
- /// </summary>
- private void InstallHandlers()
- {
- // level pack, level, difficulty handlers, characteristics
- TableView tableView = ReflectionUtil.GetPrivateField<TableView>(_beatUi.LevelPackLevelsTableView, "_tableView");
- _beatUi.LevelPackLevelsViewController.didSelectLevelEvent -= OnDidSelectLevelEvent;
- _beatUi.LevelPackLevelsViewController.didSelectLevelEvent += OnDidSelectLevelEvent;
- _beatUi.LevelDetailViewController.didPresentContentEvent -= OnDidPresentContentEvent;
- _beatUi.LevelDetailViewController.didPresentContentEvent += OnDidPresentContentEvent;
- _beatUi.LevelDetailViewController.didChangeDifficultyBeatmapEvent -= OnDidChangeDifficultyEvent;
- _beatUi.LevelDetailViewController.didChangeDifficultyBeatmapEvent += OnDidChangeDifficultyEvent;
- _beatUi.LevelPacksTableView.didSelectPackEvent -= _levelPacksTableView_didSelectPackEvent;
- _beatUi.LevelPacksTableView.didSelectPackEvent += _levelPacksTableView_didSelectPackEvent;
- _beatUi.LevelPackViewController.didSelectPackEvent -= _levelPackViewController_didSelectPackEvent;
- _beatUi.LevelPackViewController.didSelectPackEvent += _levelPackViewController_didSelectPackEvent;
- _beatUi.BeatmapCharacteristicSelectionViewController.didSelectBeatmapCharacteristicEvent -= OnDidSelectBeatmapCharacteristic;
- _beatUi.BeatmapCharacteristicSelectionViewController.didSelectBeatmapCharacteristicEvent += OnDidSelectBeatmapCharacteristic;
- // make sure the quick scroll buttons don't desync with regular scrolling
- _beatUi.TableViewPageDownButton.onClick.AddListener(delegate ()
- {
- StartCoroutine(RefreshQuickScrollButtonsAsync());
- });
- _beatUi.TableViewPageUpButton.onClick.AddListener(delegate ()
- {
- StartCoroutine(RefreshQuickScrollButtonsAsync());
- });
- // finished level
- ResultsViewController resultsViewController = _beatUi.LevelSelectionFlowCoordinator.GetPrivateField<ResultsViewController>("_resultsViewController");
- resultsViewController.continueButtonPressedEvent += ResultsViewController_continueButtonPressedEvent;
- }
- /// <summary>
- /// Helper to reduce code duplication...
- /// </summary>
- public void RefreshSongUI(bool scrollToLevel=true)
- {
- if (!_uiCreated)
- {
- return;
- }
- RefreshSongList(scrollToLevel);
- RefreshSortButtonUI();
- if (!scrollToLevel)
- {
- _beatUi.ScrollToLevelByRow(0);
- }
- RefreshQuickScrollButtons();
- RefreshCurrentSelectionDisplay();
- }
- /// <summary>
- /// External helper.
- /// </summary>
- public void ProcessSongList()
- {
- if (!_uiCreated)
- {
- return;
- }
- this._model.ProcessSongList(_beatUi.GetCurrentSelectedLevelPack());
- }
- /// <summary>
- /// Helper for common filter cancellation logic.
- /// </summary>
- public void CancelFilter()
- {
- Logger.Debug("Cancelling filter.");
- _model.Settings.filterMode = SongFilterMode.None;
- SongCore.Loader.Instance.RefreshLevelPacks();
- }
- /// <summary>
- /// Handle updating the level pack selection after returning from a song.
- /// </summary>
- /// <param name="obj"></param>
- private void ResultsViewController_continueButtonPressedEvent(ResultsViewController obj)
- {
- StartCoroutine(this.UpdateLevelPackSelectionEndOfFrame());
- }
- /// <summary>
- /// TODO - evaluate this sillyness...
- /// </summary>
- /// <returns></returns>
- public IEnumerator UpdateLevelPackSelectionEndOfFrame()
- {
- yield return new WaitForEndOfFrame();
- try
- {
- bool didUpdateLevelPack = this.UpdateLevelPackSelection();
- if (!didUpdateLevelPack)
- {
- _model.ProcessSongList(_beatUi.GetCurrentSelectedLevelPack());
- }
- _beatUi.SelectAndScrollToLevel(_beatUi.LevelPackLevelsTableView, _model.LastSelectedLevelId);
- RefreshQuickScrollButtons();
- }
- catch (Exception e)
- {
- Logger.Exception("Exception:", e);
- }
- }
- /// <summary>
- /// Handler for level pack selection.
- /// </summary>
- /// <param name="arg1"></param>
- /// <param name="arg2"></param>
- private void _levelPacksTableView_didSelectPackEvent(LevelPacksTableView arg1, IBeatmapLevelPack arg2)
- {
- Logger.Trace("_levelPacksTableView_didSelectPackEvent(arg2={0})", arg2);
- try
- {
- RefreshSortButtonUI();
- RefreshQuickScrollButtons();
- }
- catch (Exception e)
- {
- Logger.Exception("Exception handling didSelectPackEvent...", e);
- }
- }
- /// <summary>
- /// Handler for level pack selection, controller.
- /// Sets the current level pack into the model and updates.
- /// </summary>
- /// <param name="arg1"></param>
- /// <param name="arg2"></param>
- private void _levelPackViewController_didSelectPackEvent(LevelPacksViewController arg1, IBeatmapLevelPack arg2)
- {
- Logger.Trace("_levelPackViewController_didSelectPackEvent(arg2={0})", arg2);
- try
- {
- // reset filter mode always here
- if (this._model.Settings.currentLevelPackId != arg2.packID)
- {
- this._model.Settings.filterMode = SongFilterMode.None;
- }
- // save level pack
- this._model.Settings.currentLevelPackId = arg2.packID;
- this._model.Settings.Save();
- this._model.ProcessSongList(arg2);
- // trickery to handle Downloader playlist level packs
- // We need to avoid scrolling to a level and then select the header
- bool scrollToLevel = true;
- if (arg2.packID.Contains("Playlist_"))
- {
- scrollToLevel = false;
- }
- RefreshSongUI(scrollToLevel);
- }
- catch (Exception e)
- {
- Logger.Exception("Exception handling didSelectPackEvent...", e);
- }
- }
- /// <summary>
- /// Remove all filters, update song list, save.
- /// </summary>
- private void OnClearButtonClickEvent()
- {
- Logger.Debug("Clearing all sorts and filters.");
- _model.Settings.sortMode = SongSortMode.Original;
- _model.Settings.invertSortResults = false;
- _model.Settings.filterMode = SongFilterMode.None;
- _model.Settings.Save();
- SongCore.Loader.Instance.RefreshLevelPacks();
- ProcessSongList();
- RefreshSongUI();
- }
- /// <summary>
- /// Sort button clicked.
- /// </summary>
- private void OnSortButtonClickEvent(SongSortMode sortMode)
- {
- Logger.Debug("Sort button - {0} - pressed.", sortMode.ToString());
- if ((sortMode.NeedsScoreSaberData() && !SongDataCore.Plugin.ScoreSaber.IsDataAvailable()) ||
- (sortMode.NeedsBeatSaverData() && !SongDataCore.Plugin.BeatSaver.IsDataAvailable()))
- {
- Logger.Info("Data for sort type is not available.");
- return;
- }
- // Clear current selected level id so our song list jumps to the start
- _model.LastSelectedLevelId = null;
- if (_model.Settings.sortMode == sortMode)
- {
- _model.ToggleInverting();
- }
- _model.Settings.sortMode = sortMode;
- // update the seed
- if (_model.Settings.sortMode == SongSortMode.Random)
- {
- _model.Settings.randomSongSeed = Guid.NewGuid().GetHashCode();
- }
- _model.Settings.Save();
- ProcessSongList();
- RefreshSongUI();
- }
- /// <summary>
- /// Handle filter button logic. Some filters have sub menus that need special logic.
- /// </summary>
- /// <param name="mode"></param>
- private void OnFilterButtonClickEvent(SongFilterMode mode)
- {
- Logger.Debug($"FilterButton {mode} clicked.");
- // TODO - Downloader level pack support - need a way to refresh downloader level packs.
- // Every filter will lead to needing a refresh.
- SongCore.Loader.Instance.RefreshLevelPacks();
- // Always select the custom level pack - TODO - Downloader level pack support - Don't always do this
- var pack = _beatUi.GetCurrentSelectedLevelPack();
- if (pack != null && !pack.packID.Equals(PluginConfig.CUSTOM_SONG_LEVEL_PACK_ID))
- {
- _beatUi.SelectLevelPack(PluginConfig.CUSTOM_SONG_LEVEL_PACK_ID);
- }
- // If selecting the same filter, cancel
- if (_model.Settings.filterMode == mode)
- {
- _model.Settings.filterMode = SongFilterMode.None;
- }
- else
- {
- _model.Settings.filterMode = mode;
- }
- switch (mode)
- {
- case SongFilterMode.Playlist:
- OnPlaylistButtonClickEvent();
- break;
- case SongFilterMode.Search:
- OnSearchButtonClickEvent();
- break;
- default:
- _model.Settings.Save();
- ProcessSongList();
- RefreshSongUI();
- break;
- }
- }
- /// <summary>
- /// Display the keyboard.
- /// </summary>
- /// <param name="sortMode"></param>
- private void OnSearchButtonClickEvent()
- {
- Logger.Debug("Filter button - {0} - pressed.", SongFilterMode.Search.ToString());
- this.ShowSearchKeyboard();
- }
- /// <summary>
- /// Display the playlist selector.
- /// </summary>
- /// <param name="sortMode"></param>
- private void OnPlaylistButtonClickEvent()
- {
- Logger.Debug("Filter button - {0} - pressed.", SongFilterMode.Playlist.ToString());
- _model.LastSelectedLevelId = null;
- _playListFlowCoordinator.parentFlowCoordinator = _beatUi.LevelSelectionFlowCoordinator;
- _beatUi.LevelSelectionFlowCoordinator.InvokePrivateMethod("PresentFlowCoordinator", new object[] { _playListFlowCoordinator, null, false, false });
- }
- /// <summary>
- /// Adjust UI based on level selected.
- /// Various ways of detecting if a level is not properly selected. Seems most hit the first one.
- /// </summary>
- private void OnDidSelectLevelEvent(LevelPackLevelsViewController view, IPreviewBeatmapLevel level)
- {
- try
- {
- Logger.Trace("OnDidSelectLevelEvent()");
- if (level == null)
- {
- Logger.Debug("No level selected?");
- return;
- }
- if (_model.Settings == null)
- {
- Logger.Debug("Settings not instantiated yet?");
- return;
- }
- _model.LastSelectedLevelId = level.levelID;
- HandleDidSelectLevelRow(level);
- }
- catch (Exception e)
- {
- Logger.Exception("Exception selecting song:", e);
- }
- }
- /// <summary>
- /// Switching one-saber modes for example.
- /// </summary>
- /// <param name="view"></param>
- /// <param name="bc"></param>
- private void OnDidSelectBeatmapCharacteristic(BeatmapCharacteristicSegmentedControlController view, BeatmapCharacteristicSO bc)
- {
- Logger.Trace("OnDidSelectBeatmapCharacteristic({0}", bc.name);
- _model.CurrentBeatmapCharacteristicSO = bc;
- _model.UpdateLevelRecords();
- this.RefreshSongList();
- }
- /// <summary>
- /// Handle difficulty level selection.
- /// </summary>
- private void OnDidChangeDifficultyEvent(StandardLevelDetailViewController view, IDifficultyBeatmap beatmap)
- {
- Logger.Trace("OnDidChangeDifficultyEvent({0})", beatmap);
- if (view.selectedDifficultyBeatmap == null)
- {
- return;
- }
- _deleteButton.interactable = (view.selectedDifficultyBeatmap.level.levelID.Length >= 32);
- RefreshScoreSaberData(view.selectedDifficultyBeatmap.level, beatmap.difficulty);
- RefreshNoteJumpSpeed(beatmap.difficulty);
- }
- /// <summary>
- /// BeatSaber finished loading content. This is when the difficulty is finally updated.
- /// </summary>
- /// <param name="view"></param>
- /// <param name="type"></param>
- private void OnDidPresentContentEvent(StandardLevelDetailViewController view, StandardLevelDetailViewController.ContentType type)
- {
- Logger.Trace("OnDidPresentContentEvent()");
- if (view.selectedDifficultyBeatmap == null)
- {
- return;
- }
- _deleteButton.interactable = (_beatUi.LevelDetailViewController.selectedDifficultyBeatmap.level.levelID.Length >= 32);
- RefreshScoreSaberData(view.selectedDifficultyBeatmap.level, view.selectedDifficultyBeatmap.difficulty);
- RefreshNoteJumpSpeed(view.selectedDifficultyBeatmap.difficulty);
- }
- /// <summary>
- /// Refresh stats panel.
- /// </summary>
- /// <param name="level"></param>
- private void HandleDidSelectLevelRow(IPreviewBeatmapLevel level)
- {
- Logger.Trace("HandleDidSelectLevelRow({0})", level);
- _deleteButton.interactable = (level.levelID.Length >= 32);
- RefreshQuickScrollButtons();
- RefreshAddFavoriteButton(level.levelID);
- }
- /// <summary>
- /// Call Downloader delete.
- /// </summary>
- private void CallDownloaderDelete()
- {
- BeatSaverDownloader.UI.SongListTweaks.Instance.DeletePressed();
- }
- /// <summary>
- /// Pop up a delete dialog.
- /// </summary>
- private void HandleDeleteSelectedLevel()
- {
- bool DownloaderInstalled = CustomHelpers.IsModInstalled("BeatSaverDownloader");
- if (DownloaderInstalled)
- {
- CallDownloaderDelete();
- return;
- }
- IBeatmapLevel level = _beatUi.LevelDetailViewController.selectedDifficultyBeatmap.level;
- _deleteDialog.Init("Delete song", $"Do you really want to delete \"{ level.songName} {level.songSubName}\"?", "Delete", "Cancel",
- (selectedButton) =>
- {
- _beatUi.LevelSelectionFlowCoordinator.InvokePrivateMethod("DismissViewController", new object[] { _deleteDialog, null, false });
- if (selectedButton == 0)
- {
- try
- {
- // determine the index we are deleting so we can keep the cursor near the same spot after
- // the header counts as an index, so if the index came from the level array we have to add 1.
- var levelsTableView = _beatUi.LevelPackLevelsViewController.GetPrivateField<LevelPackLevelsTableView>("_levelPackLevelsTableView");
- List<IPreviewBeatmapLevel> levels = levelsTableView.GetPrivateField<IBeatmapLevelPack>("_pack").beatmapLevelCollection.beatmapLevels.ToList();
- int selectedIndex = levels.FindIndex(x => x.levelID == _beatUi.StandardLevelDetailView.selectedDifficultyBeatmap.level.levelID);
- if (selectedIndex > -1)
- {
- var song = new Song(SongCore.Loader.CustomLevels.First(x => x.Value.levelID == _beatUi.LevelDetailViewController.selectedDifficultyBeatmap.level.levelID).Value);
- SongCore.Loader.Instance.DeleteSong(song.path);
- this._model.RemoveSongFromLevelPack(_beatUi.GetCurrentSelectedLevelPack(), _beatUi.LevelDetailViewController.selectedDifficultyBeatmap.level.levelID);
- this.UpdateLevelDataModel();
- this.RefreshSongList();
- int removedLevels = levels.RemoveAll(x => x.levelID == _beatUi.StandardLevelDetailView.selectedDifficultyBeatmap.level.levelID);
- Logger.Info("Removed " + removedLevels + " level(s) from song list!");
- TableView listTableView = levelsTableView.GetPrivateField<TableView>("_tableView");
- listTableView.ScrollToCellWithIdx(selectedIndex, TableView.ScrollPositionType.Beginning, false);
- levelsTableView.SetPrivateField("_selectedRow", selectedIndex);
- listTableView.SelectCellWithIdx(selectedIndex, true);
- }
- }
- catch (Exception e)
- {
- Logger.Error("Unable to delete song! Exception: " + e);
- }
- }
- });
- _beatUi.LevelSelectionFlowCoordinator.InvokePrivateMethod("PresentViewController", new object[] { _deleteDialog, null, false });
- }
- /// <summary>
- /// Handle selection of a playlist. Show just the songs in the playlist.
- /// </summary>
- /// <param name="p"></param>
- private void HandleDidSelectPlaylist(Playlist p)
- {
- if (p != null)
- {
- Logger.Debug("Showing songs for playlist: {0}", p.playlistTitle);
- _model.Settings.filterMode = SongFilterMode.Playlist;
- _model.CurrentPlaylist = p;
- _model.Settings.Save();
- ProcessSongList();
- RefreshSongUI();
- }
- else
- {
- Logger.Debug("No playlist selected");
- }
- }
- /// <summary>
- /// Display the search keyboard
- /// </summary>
- void ShowSearchKeyboard()
- {
- if (_searchViewController == null)
- {
- _searchViewController = UIBuilder.CreateViewController<SearchKeyboardViewController>("SearchKeyboardViewController");
- _searchViewController.searchButtonPressed += SearchViewControllerSearchButtonPressed;
- _searchViewController.backButtonPressed += SearchViewControllerbackButtonPressed;
- }
- Logger.Debug("Presenting search keyboard");
- _beatUi.LevelSelectionFlowCoordinator.InvokePrivateMethod("PresentViewController", new object[] { _searchViewController, null, false });
- }
- /// <summary>
- /// Handle back button event from search keyboard.
- /// </summary>
- private void SearchViewControllerbackButtonPressed()
- {
- _beatUi.LevelSelectionFlowCoordinator.InvokePrivateMethod("DismissViewController", new object[] { _searchViewController, null, false });
- this._model.Settings.filterMode = SongFilterMode.None;
- this._model.Settings.Save();
- RefreshSongUI();
- }
- /// <summary>
- /// Handle search.
- /// </summary>
- /// <param name="searchFor"></param>
- private void SearchViewControllerSearchButtonPressed(string searchFor)
- {
- _beatUi.LevelSelectionFlowCoordinator.InvokePrivateMethod("DismissViewController", new object[] { _searchViewController, null, false });
- Logger.Debug("Searching for \"{0}\"...", searchFor);
- _model.Settings.filterMode = SongFilterMode.Search;
- _model.Settings.searchTerms.Insert(0, searchFor);
- _model.Settings.Save();
- _model.LastSelectedLevelId = null;
- ProcessSongList();
- RefreshSongUI();
- }
- /// <summary>
- /// Make big jumps in the song list.
- /// </summary>
- /// <param name="numJumps"></param>
- private void JumpSongList(int numJumps, float segmentPercent)
- {
- var levels = _beatUi.GetCurrentLevelPackLevels();
- if (levels == null)
- {
- return;
- }
- int totalSize = _beatUi.GetLevelPackLevelCount();
- int segmentSize = (int)(totalSize * segmentPercent);
- // Jump at least one scree size.
- if (segmentSize < LIST_ITEMS_VISIBLE_AT_ONCE)
- {
- segmentSize = LIST_ITEMS_VISIBLE_AT_ONCE;
- }
- TableView tableView = ReflectionUtil.GetPrivateField<TableView>(_beatUi.LevelPackLevelsTableView, "_tableView");
- int currentRow = _beatUi.LevelPackLevelsTableView.GetPrivateField<int>("_selectedRow");
- int jumpDirection = Math.Sign(numJumps);
- int newRow = currentRow + (jumpDirection * segmentSize);
- if (newRow <= 0)
- {
- newRow = 0;
- }
- else if (newRow >= totalSize)
- {
- newRow = totalSize - 1;
- }
-
- Logger.Debug("jumpDirection: {0}, newRow: {1}", jumpDirection, newRow);
- _beatUi.SelectAndScrollToLevel(_beatUi.LevelPackLevelsTableView, levels[newRow].levelID);
- RefreshQuickScrollButtons();
- }
- /// <summary>
- /// Add/Remove song from favorites depending on if it already exists.
- /// </summary>
- private void ToggleSongInPlaylist()
- {
- IBeatmapLevel songInfo = _beatUi.LevelDetailViewController.selectedDifficultyBeatmap.level;
- if (_model.CurrentEditingPlaylist != null)
- {
- if (_model.CurrentEditingPlaylistLevelIds.Contains(songInfo.levelID))
- {
- Logger.Info("Remove {0} from editing playlist", songInfo.songName);
- _model.RemoveSongFromEditingPlaylist(songInfo);
- if (_model.Settings.filterMode == SongFilterMode.Favorites)
- {
- this._model.ProcessSongList(_beatUi.GetCurrentSelectedLevelPack());
- this.RefreshSongList();
- }
- }
- else
- {
- Logger.Info("Add {0} to editing playlist", songInfo.songName);
- _model.AddSongToEditingPlaylist(songInfo);
- }
- }
- RefreshAddFavoriteButton(songInfo.levelID);
- _model.Settings.Save();
- }
- /// <summary>
- /// Update GUI elements that show score saber data.
- /// </summary>
- public void RefreshScoreSaberData(IPreviewBeatmapLevel level, BeatmapDifficulty vdifficulty)
- {
- Logger.Trace("RefreshScoreSaberData({0})", level.levelID);
- if (!SongDataCore.Plugin.ScoreSaber.IsDataAvailable())
- {
- return;
- }
- BeatmapDifficulty difficulty = _beatUi.LevelDifficultyViewController.selectedDifficulty;
- string difficultyString = difficulty.ToString();
- if (difficultyString.Equals("ExpertPlus"))
- {
- difficultyString = "Expert+";
- }
- Logger.Debug(difficultyString);
- // Check if we have data for this song
- Logger.Debug("Checking if have info for song {0}", level.songName);
- var hash = CustomHelpers.GetSongHash(level.levelID);
- if (SongDataCore.Plugin.ScoreSaber.Data.Songs.ContainsKey(hash))
- {
- Logger.Debug("Checking if have difficulty for song {0} difficulty {1}", level.songName, difficultyString);
- ScoreSaberSong scoreSaberSong = SongDataCore.Plugin.ScoreSaber.Data.Songs[hash];
- ScoreSaberSongDifficultyStats scoreSaberSongDifficulty = scoreSaberSong.diffs.FirstOrDefault(x => String.Equals(x.diff, difficultyString));
- if (scoreSaberSongDifficulty != null)
- {
- Logger.Debug("Display pp for song.");
- double pp = scoreSaberSongDifficulty.pp;
- double star = scoreSaberSongDifficulty.star;
- UIBuilder.SetStatButtonText(_ppStatButton, String.Format("{0:0.#}", pp));
- UIBuilder.SetStatButtonText(_starStatButton, String.Format("{0:0.#}", star));
- }
- else
- {
- UIBuilder.SetStatButtonText(_ppStatButton, "NA");
- UIBuilder.SetStatButtonText(_starStatButton, "NA");
- }
- }
- else
- {
- UIBuilder.SetStatButtonText(_ppStatButton, "NA");
- UIBuilder.SetStatButtonText(_starStatButton, "NA");
- }
- Logger.Debug("Done refreshing score saber stats.");
- }
- /// <summary>
- /// Helper to refresh the NJS widget.
- /// </summary>
- /// <param name="beatmap"></param>
- private void RefreshNoteJumpSpeed(BeatmapDifficulty beatmap)
- {
- UIBuilder.SetStatButtonText(_njsStatButton, String.Format("{0}", beatmap.NoteJumpMovementSpeed()));
- }
- /// <summary>
- /// Update interactive state of the quick scroll buttons.
- /// </summary>
- private void RefreshQuickScrollButtons()
- {
- if (!_uiCreated)
- {
- return;
- }
- _pageUpFastButton.interactable = _beatUi.TableViewPageUpButton.interactable;
- _pageUpFastButton.gameObject.SetActive(_beatUi.TableViewPageUpButton.IsActive());
- _pageDownFastButton.interactable = _beatUi.TableViewPageDownButton.interactable;
- _pageDownFastButton.gameObject.SetActive(_beatUi.TableViewPageDownButton.IsActive());
- }
- /// <summary>
- /// TODO - evaluate this sillyness...
- /// </summary>
- /// <returns></returns>
- public IEnumerator RefreshQuickScrollButtonsAsync()
- {
- yield return new WaitForEndOfFrame();
- RefreshQuickScrollButtons();
- }
- /// <summary>
- /// Show the UI.
- /// </summary>
- public void Show()
- {
- Logger.Trace("Show SongBrowserUI()");
- this.SetVisibility(true);
- }
- /// <summary>
- /// Hide the UI.
- /// </summary>
- public void Hide()
- {
- Logger.Trace("Hide SongBrowserUI()");
- this.SetVisibility(false);
- }
- /// <summary>
- /// Handle showing or hiding UI logic.
- /// </summary>
- /// <param name="visible"></param>
- private void SetVisibility(bool visible)
- {
- // UI not created, nothing visible to hide...
- if (!_uiCreated)
- {
- return;
- }
- _ppStatButton.gameObject.SetActive(visible);
- _starStatButton.gameObject.SetActive(visible);
- _njsStatButton.gameObject.SetActive(visible);
- RefreshOuterUIState(visible == true ? UIState.Main : UIState.Disabled);
- _addFavoriteButton.gameObject.SetActive(visible);
- _deleteButton.gameObject.SetActive(visible);
- _pageUpFastButton.gameObject.SetActive(visible);
- _pageDownFastButton.gameObject.SetActive(visible);
- }
- /// <summary>
- /// Update the top UI state.
- /// Hides the outer ui, sort, and filter buttons depending on the state.
- /// </summary>
- private void RefreshOuterUIState(UIState state)
- {
- bool sortButtons = false;
- bool filterButtons = false;
- bool outerButtons = false;
- bool clearButton = true;
- if (state == UIState.SortBy)
- {
- sortButtons = true;
- }
- else if (state == UIState.FilterBy)
- {
- filterButtons = true;
- }
- else if (state == UIState.Main)
- {
- outerButtons = true;
- }
- else
- {
- clearButton = false;
- }
- _sortButtonGroup.ForEach(x => x.Button.gameObject.SetActive(sortButtons));
- _filterButtonGroup.ForEach(x => x.Button.gameObject.SetActive(filterButtons));
- _sortByButton.gameObject.SetActive(outerButtons);
- _sortByDisplay.gameObject.SetActive(outerButtons);
- _filterByButton.gameObject.SetActive(outerButtons);
- _filterByDisplay.gameObject.SetActive(outerButtons);
- _clearSortFilterButton.gameObject.SetActive(clearButton);
- _randomButton.gameObject.SetActive(outerButtons);
- RefreshCurrentSelectionDisplay();
- _currentUiState = state;
- }
- /// <summary>
- /// Adjust the text field of the sort by and filter by displays.
- /// </summary>
- private void RefreshCurrentSelectionDisplay()
- {
- string sortByDisplay = null;
- if (_model.Settings.sortMode == SongSortMode.Default)
- {
- sortByDisplay = "Title";
- }
- else
- {
- sortByDisplay = _model.Settings.sortMode.ToString();
- }
- _sortByDisplay.SetButtonText(sortByDisplay);
- if (_model.Settings.filterMode != SongFilterMode.Custom)
- {
- // Custom SongFilterMod implies that another mod has modified the text of this button (do not overwrite)
- _filterByDisplay.SetButtonText(_model.Settings.filterMode.ToString());
- }
- }
- /// <summary>
- /// Helper to quickly refresh add to favorites button
- /// </summary>
- /// <param name="levelId"></param>
- private void RefreshAddFavoriteButton(String levelId)
- {
- if (levelId == null)
- {
- _currentAddFavoriteButtonSprite = null;
- }
- else
- {
- if (_model.CurrentEditingPlaylistLevelIds.Contains(levelId))
- {
- _currentAddFavoriteButtonSprite = Base64Sprites.RemoveFromFavoritesIcon;
- }
- else
- {
- _currentAddFavoriteButtonSprite = Base64Sprites.AddToFavoritesIcon;
- }
- }
- _addFavoriteButton.SetButtonIcon(_currentAddFavoriteButtonSprite);
- }
- /// <summary>
- /// Adjust the UI colors.
- /// </summary>
- public void RefreshSortButtonUI()
- {
- if (!_uiCreated)
- {
- return;
- }
- // So far all we need to refresh is the sort buttons.
- foreach (SongSortButton sortButton in _sortButtonGroup)
- {
- if (sortButton.SortMode.NeedsBeatSaverData() && !SongDataCore.Plugin.BeatSaver.IsDataAvailable())
- {
- UIBuilder.SetButtonBorder(sortButton.Button, Color.gray);
- }
- else if (sortButton.SortMode.NeedsScoreSaberData() && !SongDataCore.Plugin.ScoreSaber.IsDataAvailable())
- {
- UIBuilder.SetButtonBorder(sortButton.Button, Color.gray);
- }
- else
- {
- UIBuilder.SetButtonBorder(sortButton.Button, Color.white);
- }
- if (sortButton.SortMode == _model.Settings.sortMode)
- {
- if (this._model.Settings.invertSortResults)
- {
- UIBuilder.SetButtonBorder(sortButton.Button, Color.red);
- }
- else
- {
- UIBuilder.SetButtonBorder(sortButton.Button, Color.green);
- }
- }
- }
- // refresh filter buttons
- foreach (SongFilterButton filterButton in _filterButtonGroup)
- {
- UIBuilder.SetButtonBorder(filterButton.Button, Color.white);
- if (filterButton.FilterMode == _model.Settings.filterMode)
- {
- UIBuilder.SetButtonBorder(filterButton.Button, Color.green);
- }
- }
- if (this._model.Settings.invertSortResults)
- {
- UIBuilder.SetButtonBorder(_sortByDisplay, Color.red);
- }
- else
- {
- UIBuilder.SetButtonBorder(_sortByDisplay, Color.green);
- }
- if (this._model.Settings.filterMode != SongFilterMode.None)
- {
- UIBuilder.SetButtonBorder(_filterByDisplay, Color.green);
- }
- else
- {
- UIBuilder.SetButtonBorder(_filterByDisplay, Color.white);
- }
- }
- /// <summary>
- ///
- /// </summary>
- public void RefreshSongList(bool scrollToLevel = true)
- {
- if (!_uiCreated)
- {
- return;
- }
- _beatUi.RefreshSongList(_model.LastSelectedLevelId);
- }
- /// <summary>
- /// Helper for updating the model (which updates the song list)
- /// </summary>
- public void UpdateLevelDataModel()
- {
- try
- {
- Logger.Trace("UpdateLevelDataModel()");
- // get a current beatmap characteristic...
- if (_model.CurrentBeatmapCharacteristicSO == null && _uiCreated)
- {
- _model.CurrentBeatmapCharacteristicSO = _beatUi.BeatmapCharacteristicSelectionViewController.GetPrivateField<BeatmapCharacteristicSO>("_selectedBeatmapCharacteristic");
- }
- _model.UpdateLevelRecords();
- bool didUpdateLevelPack = UpdateLevelPackSelection();
- if (didUpdateLevelPack)
- {
- ProcessSongList();
- }
- }
- catch (Exception e)
- {
- Logger.Exception("SongBrowser UI crashed trying to update the internal song lists: ", e);
- }
- }
- /// <summary>
- /// Logic for fixing BeatSaber's level pack selection bugs.
- /// </summary>
- public bool UpdateLevelPackSelection()
- {
- if (_uiCreated)
- {
- IBeatmapLevelPack currentSelected = _beatUi.GetCurrentSelectedLevelPack();
- Logger.Debug("Current selected level pack: {0}", currentSelected);
- if (String.IsNullOrEmpty(_model.Settings.currentLevelPackId))
- {
- if (currentSelected == null)
- {
- Logger.Debug("No level pack selected, acquiring the first available...");
- var levelPackCollection = _beatUi.LevelPackViewController.GetPrivateField<IBeatmapLevelPackCollection>("_levelPackCollection");
- currentSelected = levelPackCollection.beatmapLevelPacks[0];
- }
- }
- else if (currentSelected == null || (currentSelected.packID != _model.Settings.currentLevelPackId))
- {
- Logger.Debug("Automatically selecting level pack: {0}", _model.Settings.currentLevelPackId);
- // HACK - BeatSaber seems to always go back to OST1 internally.
- // - Lets force it to the last pack id but not have SongBrowser functions fire.
- // Turn off our event processing
- _beatUi.LevelPackViewController.didSelectPackEvent -= _levelPackViewController_didSelectPackEvent;
- _beatUi.LevelPacksTableView.didSelectPackEvent -= _levelPacksTableView_didSelectPackEvent;
- var levelPack = _beatUi.GetLevelPackByPackId(_model.Settings.currentLevelPackId);
- _beatUi.SelectLevelPack(_model.Settings.currentLevelPackId);
- ProcessSongList();
- _beatUi.LevelPackViewController.didSelectPackEvent += _levelPackViewController_didSelectPackEvent;
- _beatUi.LevelPacksTableView.didSelectPackEvent += _levelPacksTableView_didSelectPackEvent;
-
- return true;
- }
- }
- return false;
- }
- }
- }
-
|