osu/osu.Game/Overlays/Settings/Sections/SkinSection.cs

192 lines
6.8 KiB
C#
Raw Normal View History

// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
// See the LICENCE file in the repository root for full licence text.
2018-04-13 09:19:50 +00:00
2020-05-19 07:44:22 +00:00
using System;
2018-11-28 11:36:21 +00:00
using System.Collections.Generic;
2018-04-13 09:19:50 +00:00
using System.Linq;
using osu.Framework.Allocation;
2019-02-21 10:04:31 +00:00
using osu.Framework.Bindables;
2018-04-13 09:19:50 +00:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Sprites;
2020-05-24 04:44:11 +00:00
using osu.Framework.Logging;
2018-04-13 09:19:50 +00:00
using osu.Game.Configuration;
using osu.Game.Graphics.UserInterface;
using osu.Game.Skinning;
2018-11-20 07:51:59 +00:00
using osuTK;
2018-04-13 09:19:50 +00:00
namespace osu.Game.Overlays.Settings.Sections
{
public class SkinSection : SettingsSection
{
2018-11-14 09:02:38 +00:00
private SkinSettingsDropdown skinDropdown;
2018-04-13 09:19:50 +00:00
public override string Header => "Skin";
public override Drawable CreateIcon() => new SpriteIcon
{
Icon = FontAwesome.Solid.PaintBrush
};
2018-04-13 09:19:50 +00:00
private readonly Bindable<SkinInfo> dropdownBindable = new Bindable<SkinInfo> { Default = SkinInfo.Default };
2018-11-14 09:02:38 +00:00
private readonly Bindable<int> configBindable = new Bindable<int>();
private static readonly SkinInfo random_skin_info = new SkinInfo
{
ID = SkinInfo.RANDOM_SKIN,
Name = "<Random Skin>",
};
2018-11-25 02:50:26 +00:00
private List<SkinInfo> skinItems;
2018-04-13 09:19:50 +00:00
2021-01-19 14:00:17 +00:00
private int firstNonDefault;
2020-02-14 13:14:00 +00:00
[Resolved]
private SkinManager skins { get; set; }
2018-04-13 09:19:50 +00:00
private IBindable<WeakReference<SkinInfo>> managerUpdated;
2020-05-19 07:44:22 +00:00
private IBindable<WeakReference<SkinInfo>> managerRemoved;
2018-04-13 09:19:50 +00:00
[BackgroundDependencyLoader]
2020-02-14 13:14:00 +00:00
private void load(OsuConfigManager config)
2018-04-13 09:19:50 +00:00
{
FlowContent.Spacing = new Vector2(0, 5);
2020-05-24 04:44:11 +00:00
2018-04-13 09:19:50 +00:00
Children = new Drawable[]
{
2018-11-14 09:02:38 +00:00
skinDropdown = new SkinSettingsDropdown(),
2020-05-24 14:15:24 +00:00
new ExportSkinButton(),
2019-09-02 22:28:51 +00:00
new SettingsSlider<float, SizeSlider>
2018-04-13 09:19:50 +00:00
{
LabelText = "Gameplay cursor size",
Current = config.GetBindable<float>(OsuSetting.GameplayCursorSize),
2018-07-10 15:28:56 +00:00
KeyboardStep = 0.01f
2018-04-13 09:19:50 +00:00
},
new SettingsCheckbox
{
LabelText = "Adjust gameplay cursor size based on current beatmap",
Current = config.GetBindable<bool>(OsuSetting.AutoCursorSize)
2018-04-13 09:19:50 +00:00
},
new SettingsCheckbox
2019-01-21 14:04:06 +00:00
{
LabelText = "Beatmap skins",
Current = config.GetBindable<bool>(OsuSetting.BeatmapSkins)
2019-01-21 14:04:06 +00:00
},
new SettingsCheckbox
{
LabelText = "Beatmap colours",
Current = config.GetBindable<bool>(OsuSetting.BeatmapColours)
},
new SettingsCheckbox
2018-04-13 09:19:50 +00:00
{
LabelText = "Beatmap hitsounds",
Current = config.GetBindable<bool>(OsuSetting.BeatmapHitsounds)
2018-04-13 09:19:50 +00:00
},
};
managerUpdated = skins.ItemUpdated.GetBoundCopy();
managerUpdated.BindValueChanged(itemUpdated);
2018-04-13 09:19:50 +00:00
2020-05-19 07:44:22 +00:00
managerRemoved = skins.ItemRemoved.GetBoundCopy();
managerRemoved.BindValueChanged(itemRemoved);
2018-04-13 09:19:50 +00:00
2018-11-14 09:02:38 +00:00
config.BindWith(OsuSetting.Skin, configBindable);
2018-11-25 02:50:26 +00:00
skinDropdown.Current = dropdownBindable;
updateItems();
// Todo: This should not be necessary when OsuConfigManager is databased
2018-11-14 10:29:20 +00:00
if (skinDropdown.Items.All(s => s.ID != configBindable.Value))
2018-11-14 09:02:38 +00:00
configBindable.Value = 0;
configBindable.BindValueChanged(id => dropdownBindable.Value = skinDropdown.Items.Single(s => s.ID == id.NewValue), true);
dropdownBindable.BindValueChanged(skin =>
2018-11-25 02:50:26 +00:00
{
if (skin.NewValue == random_skin_info)
{
2020-11-11 04:05:03 +00:00
skins.SelectRandomSkin();
return;
}
configBindable.Value = skin.NewValue.ID;
2018-11-25 02:50:26 +00:00
});
}
private void updateItems()
2018-11-28 11:36:21 +00:00
{
skinItems = skins.GetAllUsableSkins();
2021-01-19 14:00:17 +00:00
firstNonDefault = skinItems.FindIndex(s => s.ID > 0);
2021-01-19 16:55:50 +00:00
if (firstNonDefault < 0)
firstNonDefault = skinItems.Count;
2021-01-19 14:00:17 +00:00
skinItems.Insert(firstNonDefault, random_skin_info);
2021-01-18 19:51:42 +00:00
skinItems = sortList(skinItems);
skinDropdown.Items = skinItems;
}
2018-11-28 11:36:21 +00:00
private void itemUpdated(ValueChangedEvent<WeakReference<SkinInfo>> weakItem)
2018-11-28 11:36:21 +00:00
{
2020-05-19 07:44:22 +00:00
if (weakItem.NewValue.TryGetTarget(out var item))
2021-01-18 19:51:42 +00:00
{
2021-01-19 12:17:56 +00:00
List<SkinInfo> newDropdownItems = skinDropdown.Items.Where(i => !i.Equals(item)).Append(item).ToList();
2021-01-18 19:51:42 +00:00
newDropdownItems = sortList(newDropdownItems);
Schedule(() => skinDropdown.Items = newDropdownItems.ToArray());
}
2018-11-28 11:36:21 +00:00
}
2018-04-13 09:19:50 +00:00
2020-05-19 07:44:22 +00:00
private void itemRemoved(ValueChangedEvent<WeakReference<SkinInfo>> weakItem)
2018-04-13 09:19:50 +00:00
{
2020-05-19 07:44:22 +00:00
if (weakItem.NewValue.TryGetTarget(out var item))
Schedule(() => skinDropdown.Items = skinDropdown.Items.Where(i => i.ID != item.ID).ToArray());
2018-04-13 09:19:50 +00:00
}
2021-01-18 19:51:42 +00:00
private List<SkinInfo> sortList(List<SkinInfo> skinsList)
{
2021-01-19 14:00:17 +00:00
// Sort user skins seperate from built-in skins
List<SkinInfo> userSkinsList = skinsList.GetRange(firstNonDefault + 1, skinsList.Count - (firstNonDefault + 1));
skinsList.RemoveRange(firstNonDefault + 1, skinsList.Count - (firstNonDefault + 1));
2021-01-19 16:11:16 +00:00
userSkinsList.Sort((a, b) => string.Compare(a.Name, b.Name, StringComparison.OrdinalIgnoreCase));
2021-01-19 14:00:17 +00:00
skinsList.AddRange(userSkinsList);
2021-01-18 19:51:42 +00:00
return skinsList;
}
2018-11-14 10:29:20 +00:00
private class SkinSettingsDropdown : SettingsDropdown<SkinInfo>
2018-11-14 09:02:38 +00:00
{
protected override OsuDropdown<SkinInfo> CreateDropdown() => new SkinDropdownControl();
2018-11-14 09:02:38 +00:00
private class SkinDropdownControl : DropdownControl
{
2018-11-14 10:29:20 +00:00
protected override string GenerateItemText(SkinInfo item) => item.ToString();
2018-11-14 09:02:38 +00:00
}
}
2018-11-25 02:50:26 +00:00
2020-05-24 14:15:24 +00:00
private class ExportSkinButton : SettingsButton
{
[Resolved]
private SkinManager skins { get; set; }
private Bindable<Skin> currentSkin;
[BackgroundDependencyLoader]
private void load()
{
Text = "Export selected skin";
Action = export;
currentSkin = skins.CurrentSkin.GetBoundCopy();
currentSkin.BindValueChanged(skin => Enabled.Value = skin.NewValue.SkinInfo.ID > 0, true);
}
private void export()
{
try
{
skins.Export(currentSkin.Value.SkinInfo);
}
catch (Exception e)
{
Logger.Log($"Could not export current skin: {e.Message}", level: LogLevel.Error);
}
}
}
2018-04-13 09:19:50 +00:00
}
}