mirror of
https://github.com/NotAKidoS/NAK_CVR_Mods.git
synced 2025-08-31 13:29:27 +00:00
[Tinyboard] Initial release
This commit is contained in:
parent
c368daab4f
commit
6bef1d0c96
5 changed files with 426 additions and 0 deletions
346
Tinyboard/Main.cs
Normal file
346
Tinyboard/Main.cs
Normal file
|
@ -0,0 +1,346 @@
|
|||
using System.Reflection;
|
||||
using ABI_RC.Core.InteractionSystem;
|
||||
using ABI_RC.Core.Savior;
|
||||
using ABI_RC.Core.UI;
|
||||
using ABI_RC.Core.UI.UIRework.Managers;
|
||||
using ABI_RC.Systems.VRModeSwitch;
|
||||
using ABI_RC.VideoPlayer.Scripts;
|
||||
using HarmonyLib;
|
||||
using MelonLoader;
|
||||
using TMPro;
|
||||
using UnityEngine;
|
||||
using UnityEngine.UI;
|
||||
|
||||
namespace NAK.Tinyboard;
|
||||
|
||||
public class TinyboardMod : MelonMod
|
||||
{
|
||||
#region Melon Preferences
|
||||
|
||||
private static readonly MelonPreferences_Category Category =
|
||||
MelonPreferences.CreateCategory(nameof(Tinyboard));
|
||||
|
||||
private static readonly MelonPreferences_Entry<bool> EntrySmartAlignToMenu =
|
||||
Category.CreateEntry(
|
||||
identifier: "smart_align_to_menu",
|
||||
true,
|
||||
display_name: "Smart Align To Menu",
|
||||
description: "Should the keyboard align to the menu it was opened from? (Main Menu, World-Anchored Quick Menu)");
|
||||
|
||||
private static readonly MelonPreferences_Entry<bool> EntryEnforceTitle =
|
||||
Category.CreateEntry(
|
||||
identifier: "enforce_title",
|
||||
true,
|
||||
display_name: "Enforce Title",
|
||||
description: "Should the keyboard enforce a title when opened from an input field or main menu?");
|
||||
|
||||
private static readonly MelonPreferences_Entry<bool> EntryResizeKeyboard =
|
||||
Category.CreateEntry(
|
||||
identifier: "resize_keyboard",
|
||||
true,
|
||||
display_name: "Resize Keyboard",
|
||||
description: "Should the keyboard be resized to match XSOverlays width?");
|
||||
|
||||
private static readonly MelonPreferences_Entry<bool> EntryUseModifiers =
|
||||
Category.CreateEntry(
|
||||
identifier: "use_scale_distance_modifiers",
|
||||
true,
|
||||
display_name: "Use Scale/Distance/Offset Modifiers",
|
||||
description: "Should the scale/distance/offset modifiers be used?");
|
||||
|
||||
private static readonly MelonPreferences_Entry<float> EntryDesktopScaleModifier =
|
||||
Category.CreateEntry(
|
||||
identifier: "desktop_scale_modifier",
|
||||
0.75f,
|
||||
display_name: "Desktop Scale Modifier",
|
||||
description: "Scale modifier for desktop mode.");
|
||||
|
||||
private static readonly MelonPreferences_Entry<float> EntryDesktopDistance =
|
||||
Category.CreateEntry(
|
||||
identifier: "desktop_distance_modifier",
|
||||
0f,
|
||||
display_name: "Desktop Distance Modifier",
|
||||
description: "Distance modifier for desktop mode.");
|
||||
|
||||
private static readonly MelonPreferences_Entry<float> EntryDesktopVerticalAdjustment =
|
||||
Category.CreateEntry(
|
||||
identifier: "desktop_vertical_adjustment",
|
||||
0.1f,
|
||||
display_name: "Desktop Vertical Adjustment",
|
||||
description: "Vertical adjustment for desktop mode.");
|
||||
|
||||
private static readonly MelonPreferences_Entry<float> EntryVRScaleModifier =
|
||||
Category.CreateEntry(
|
||||
identifier: "vr_scale_modifier",
|
||||
0.85f,
|
||||
display_name: "VR Scale Modifier",
|
||||
description: "Scale modifier for VR mode.");
|
||||
|
||||
private static readonly MelonPreferences_Entry<float> EntryVRDistance =
|
||||
Category.CreateEntry(
|
||||
identifier: "vr_distance_modifier",
|
||||
0.2f,
|
||||
display_name: "VR Distance Modifier",
|
||||
description: "Distance modifier for VR mode.");
|
||||
|
||||
private static readonly MelonPreferences_Entry<float> EntryVRVerticalAdjustment =
|
||||
Category.CreateEntry(
|
||||
identifier: "vr_vertical_adjustment",
|
||||
0f,
|
||||
display_name: "VR Vertical Adjustment",
|
||||
description: "Vertical adjustment for VR mode.");
|
||||
|
||||
#endregion Melon Preferences
|
||||
|
||||
private static Transform _tinyBoardOffset;
|
||||
private static void ApplyTinyBoardOffsetsForVRMode()
|
||||
{
|
||||
if (!EntryUseModifiers.Value)
|
||||
{
|
||||
_tinyBoardOffset.localScale = Vector3.one;
|
||||
_tinyBoardOffset.localPosition = Vector3.zero;
|
||||
return;
|
||||
}
|
||||
float distanceModifier;
|
||||
float scaleModifier;
|
||||
float verticalAdjustment;
|
||||
if (MetaPort.Instance.isUsingVr)
|
||||
{
|
||||
scaleModifier = EntryVRScaleModifier.Value;
|
||||
distanceModifier = EntryVRDistance.Value;
|
||||
verticalAdjustment = EntryVRVerticalAdjustment.Value;
|
||||
}
|
||||
else
|
||||
{
|
||||
scaleModifier = EntryDesktopScaleModifier.Value;
|
||||
distanceModifier = EntryDesktopDistance.Value;
|
||||
verticalAdjustment = EntryDesktopVerticalAdjustment.Value;
|
||||
}
|
||||
_tinyBoardOffset.localScale = Vector3.one * scaleModifier;
|
||||
_tinyBoardOffset.localPosition = new Vector3(0f, verticalAdjustment, distanceModifier);
|
||||
}
|
||||
|
||||
private static void ApplyTinyBoardWidthResize()
|
||||
{
|
||||
KeyboardManager km = KeyboardManager.Instance;
|
||||
CohtmlControlledView cohtmlView = km.cohtmlView;
|
||||
Transform keyboardTransform = cohtmlView.transform;
|
||||
|
||||
int targetWidthPixels = EntryResizeKeyboard.Value ? 1330 : 1520;
|
||||
float targetScaleX = EntryResizeKeyboard.Value ? 1.4f : 1.6f;
|
||||
|
||||
cohtmlView.Width = targetWidthPixels;
|
||||
Vector3 currentScale = keyboardTransform.localScale;
|
||||
currentScale.x = targetScaleX;
|
||||
keyboardTransform.localScale = currentScale;
|
||||
}
|
||||
|
||||
public override void OnInitializeMelon()
|
||||
{
|
||||
// add our shim transform to scale the menu down by 0.75
|
||||
HarmonyInstance.Patch(
|
||||
typeof(CVRKeyboardPositionHelper).GetMethod(nameof(CVRKeyboardPositionHelper.Awake),
|
||||
BindingFlags.NonPublic | BindingFlags.Instance),
|
||||
postfix: new HarmonyMethod(typeof(TinyboardMod).GetMethod(nameof(OnCVRKeyboardPositionHelperAwake),
|
||||
BindingFlags.NonPublic | BindingFlags.Static))
|
||||
);
|
||||
|
||||
// reposition the keyboard when it is opened to match the menu position if it is opened from a menu
|
||||
HarmonyInstance.Patch(
|
||||
typeof(MenuPositionHelperBase).GetMethod(nameof(MenuPositionHelperBase.OnMenuOpen),
|
||||
BindingFlags.NonPublic | BindingFlags.Instance),
|
||||
postfix: new HarmonyMethod(typeof(TinyboardMod).GetMethod(nameof(OnMenuPositionHelperBaseOnMenuOpen),
|
||||
BindingFlags.NonPublic | BindingFlags.Static))
|
||||
);
|
||||
|
||||
// enforces a title for the keyboard in cases it did not already have one
|
||||
HarmonyInstance.Patch(
|
||||
typeof(KeyboardManager).GetMethod(nameof(KeyboardManager.ShowKeyboard),
|
||||
BindingFlags.Public | BindingFlags.Instance),
|
||||
prefix: new HarmonyMethod(typeof(TinyboardMod).GetMethod(nameof(OnKeyboardManagerShowKeyboard),
|
||||
BindingFlags.NonPublic | BindingFlags.Static))
|
||||
);
|
||||
|
||||
// resize keyboard to match XSOverlays width
|
||||
HarmonyInstance.Patch(
|
||||
typeof(KeyboardManager).GetMethod(nameof(KeyboardManager.Start),
|
||||
BindingFlags.NonPublic | BindingFlags.Instance),
|
||||
postfix: new HarmonyMethod(typeof(TinyboardMod).GetMethod(nameof(OnKeyboardManagerStart),
|
||||
BindingFlags.NonPublic | BindingFlags.Static))
|
||||
);
|
||||
|
||||
// update offsets when switching VR modes
|
||||
VRModeSwitchEvents.OnPostVRModeSwitch.AddListener((_) => ApplyTinyBoardOffsetsForVRMode());
|
||||
|
||||
// listen for setting changes
|
||||
EntryUseModifiers.OnEntryValueChanged.Subscribe((_,_) => ApplyTinyBoardOffsetsForVRMode());
|
||||
EntryDesktopScaleModifier.OnEntryValueChanged.Subscribe((_,_) => ApplyTinyBoardOffsetsForVRMode());
|
||||
EntryVRScaleModifier.OnEntryValueChanged.Subscribe((_,_) => ApplyTinyBoardOffsetsForVRMode());
|
||||
EntryDesktopDistance.OnEntryValueChanged.Subscribe((_,_) => ApplyTinyBoardOffsetsForVRMode());
|
||||
EntryVRDistance.OnEntryValueChanged.Subscribe((_,_) => ApplyTinyBoardOffsetsForVRMode());
|
||||
EntryResizeKeyboard.OnEntryValueChanged.Subscribe((_,_) => ApplyTinyBoardWidthResize());
|
||||
}
|
||||
|
||||
private static void OnCVRKeyboardPositionHelperAwake(CVRKeyboardPositionHelper __instance)
|
||||
{
|
||||
_tinyBoardOffset = new GameObject("NAKTinyBoard").transform;
|
||||
|
||||
Transform offsetTransform = __instance.transform.GetChild(0);
|
||||
_tinyBoardOffset.SetParent(offsetTransform, false);
|
||||
|
||||
ApplyTinyBoardOffsetsForVRMode();
|
||||
|
||||
Transform menuTransform = __instance.menuTransform;
|
||||
menuTransform.SetParent(_tinyBoardOffset, false);
|
||||
}
|
||||
|
||||
private static void OnMenuPositionHelperBaseOnMenuOpen(MenuPositionHelperBase __instance)
|
||||
{
|
||||
if (!EntrySmartAlignToMenu.Value) return;
|
||||
if (__instance is not CVRKeyboardPositionHelper { IsMenuOpen: true }) return;
|
||||
|
||||
// Check if the open source was an open menu
|
||||
KeyboardManager.OpenSource? openSource = KeyboardManager.Instance._keyboardOpenSource;
|
||||
|
||||
MenuPositionHelperBase menuPositionHelper;
|
||||
switch (openSource)
|
||||
{
|
||||
case KeyboardManager.OpenSource.MainMenu:
|
||||
menuPositionHelper = CVRMainMenuPositionHelper.Instance;
|
||||
break;
|
||||
case KeyboardManager.OpenSource.QuickMenu:
|
||||
menuPositionHelper = CVRQuickMenuPositionHelper.Instance;
|
||||
if (!menuPositionHelper.IsUsingWorldAnchoredMenu) return; // hand anchored quick menu, don't touch
|
||||
break;
|
||||
default: return;
|
||||
}
|
||||
|
||||
// get modifiers
|
||||
float rootScaleModifier = __instance.transform.lossyScale.x;
|
||||
float keyboardDistanceModifier = __instance.MenuDistanceModifier;
|
||||
float menuDistanceModifier = menuPositionHelper.MenuDistanceModifier;
|
||||
|
||||
// get difference between modifiers
|
||||
float distanceModifier = keyboardDistanceModifier - menuDistanceModifier;
|
||||
|
||||
// place keyboard at menu position + difference in modifiers
|
||||
Transform menuOffsetTransform = menuPositionHelper._offsetTransform;
|
||||
Quaternion keyboardRotation = menuOffsetTransform.rotation;
|
||||
Vector3 keyboardPosition = menuOffsetTransform.position +
|
||||
menuOffsetTransform.forward * (rootScaleModifier * distanceModifier);
|
||||
|
||||
// place keyboard as if it was opened with player camera in same place as menu was
|
||||
__instance._offsetTransform.SetPositionAndRotation(keyboardPosition, keyboardRotation);
|
||||
}
|
||||
|
||||
private static void OnKeyboardManagerStart() => ApplyTinyBoardWidthResize();
|
||||
|
||||
/*
|
||||
public void ShowKeyboard(
|
||||
string currentText,
|
||||
Action<string> callback,
|
||||
string placeholder = null,
|
||||
string successText = "Success",
|
||||
int maxCharacterCount = 0,
|
||||
bool hidden = false,
|
||||
bool multiLine = false,
|
||||
string title = null,
|
||||
OpenSource openSource = OpenSource.Other)
|
||||
*/
|
||||
|
||||
// using mix of index and args params because otherwise explodes with invalid IL ?
|
||||
private static void OnKeyboardManagerShowKeyboard(ref string __7, ref string __2, object[] __args)
|
||||
{
|
||||
if (!EntryEnforceTitle.Value) return;
|
||||
|
||||
// ReSharper disable thrice InlineTemporaryVariable
|
||||
ref string title = ref __7;
|
||||
ref string placeholder = ref __2;
|
||||
if (!string.IsNullOrWhiteSpace(title)) return;
|
||||
|
||||
Action<string> callback = __args[1] as Action<string>;
|
||||
KeyboardManager.OpenSource? openSource = __args[8] as KeyboardManager.OpenSource?;
|
||||
|
||||
if (callback?.Target != null)
|
||||
{
|
||||
var target = callback.Target;
|
||||
switch (openSource)
|
||||
{
|
||||
case KeyboardManager.OpenSource.CVRInputFieldKeyboardHandler:
|
||||
TrySetPlaceholderFromKeyboardHandler(target, ref title, ref placeholder);
|
||||
break;
|
||||
case KeyboardManager.OpenSource.MainMenu:
|
||||
title = TryExtractTitleFromMainMenu(target);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!string.IsNullOrWhiteSpace(placeholder))
|
||||
{
|
||||
// fallback to placeholder if no title found
|
||||
if (string.IsNullOrWhiteSpace(title)) title = placeholder;
|
||||
|
||||
// clear placeholder if it is longer than 10 characters
|
||||
if (placeholder.Length > 10) placeholder = string.Empty;
|
||||
}
|
||||
}
|
||||
|
||||
private static void TrySetPlaceholderFromKeyboardHandler(object target, ref string title, ref string placeholder)
|
||||
{
|
||||
Type type = target.GetType();
|
||||
|
||||
TMP_InputField tmpInput = type.GetField("input", BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic)?.GetValue(target) as TMP_InputField;
|
||||
if (tmpInput != null)
|
||||
{
|
||||
if (tmpInput.GetComponentInParent<ViewManagerVideoPlayer>()) title = "VideoPlayer URL or Search";
|
||||
if (tmpInput.placeholder is TMP_Text ph)
|
||||
{
|
||||
placeholder = ph.text;
|
||||
return;
|
||||
}
|
||||
placeholder = PrettyString(tmpInput.gameObject.name);
|
||||
return;
|
||||
}
|
||||
|
||||
InputField legacyInput = type.GetField("inputField", BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic)?.GetValue(target) as InputField;
|
||||
if (legacyInput != null)
|
||||
{
|
||||
if (legacyInput.placeholder is Text ph)
|
||||
{
|
||||
placeholder = ph.text;
|
||||
return;
|
||||
}
|
||||
placeholder = PrettyString(legacyInput.gameObject.name);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
private static string TryExtractTitleFromMainMenu(object target)
|
||||
{
|
||||
Type type = target.GetType();
|
||||
string targetId = type.GetField("targetId", BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic)?.GetValue(target) as string;
|
||||
return string.IsNullOrWhiteSpace(targetId) ? null : PrettyString(targetId);
|
||||
}
|
||||
|
||||
private static string PrettyString(string str)
|
||||
{
|
||||
int len = str.Length;
|
||||
Span<char> buffer = stackalloc char[len * 2];
|
||||
int pos = 0;
|
||||
bool newWord = true;
|
||||
for (int i = 0; i < len; i++)
|
||||
{
|
||||
char c = str[i];
|
||||
if (c is '_' or '-')
|
||||
{
|
||||
buffer[pos++] = ' ';
|
||||
newWord = true;
|
||||
continue;
|
||||
}
|
||||
if (char.IsUpper(c) && i > 0 && !newWord) buffer[pos++] = ' ';
|
||||
buffer[pos++] = newWord ? char.ToUpperInvariant(c) : c;
|
||||
newWord = false;
|
||||
}
|
||||
return new string(buffer[..pos]);
|
||||
}
|
||||
}
|
32
Tinyboard/Properties/AssemblyInfo.cs
Normal file
32
Tinyboard/Properties/AssemblyInfo.cs
Normal file
|
@ -0,0 +1,32 @@
|
|||
using MelonLoader;
|
||||
using NAK.Tinyboard.Properties;
|
||||
using System.Reflection;
|
||||
|
||||
[assembly: AssemblyVersion(AssemblyInfoParams.Version)]
|
||||
[assembly: AssemblyFileVersion(AssemblyInfoParams.Version)]
|
||||
[assembly: AssemblyInformationalVersion(AssemblyInfoParams.Version)]
|
||||
[assembly: AssemblyTitle(nameof(NAK.Tinyboard))]
|
||||
[assembly: AssemblyCompany(AssemblyInfoParams.Author)]
|
||||
[assembly: AssemblyProduct(nameof(NAK.Tinyboard))]
|
||||
|
||||
[assembly: MelonInfo(
|
||||
typeof(NAK.Tinyboard.TinyboardMod),
|
||||
nameof(NAK.Tinyboard),
|
||||
AssemblyInfoParams.Version,
|
||||
AssemblyInfoParams.Author,
|
||||
downloadLink: "https://github.com/NotAKidoS/NAK_CVR_Mods/tree/main/Tinyboard"
|
||||
)]
|
||||
|
||||
[assembly: MelonGame("ChilloutVR", "ChilloutVR")]
|
||||
[assembly: MelonPlatform(MelonPlatformAttribute.CompatiblePlatforms.WINDOWS_X64)]
|
||||
[assembly: MelonPlatformDomain(MelonPlatformDomainAttribute.CompatibleDomains.MONO)]
|
||||
[assembly: MelonColor(255, 246, 25, 99)] // red-pink
|
||||
[assembly: MelonAuthorColor(255, 158, 21, 32)] // red
|
||||
[assembly: HarmonyDontPatchAll]
|
||||
|
||||
namespace NAK.Tinyboard.Properties;
|
||||
internal static class AssemblyInfoParams
|
||||
{
|
||||
public const string Version = "1.0.0";
|
||||
public const string Author = "NotAKidoS";
|
||||
}
|
19
Tinyboard/README.md
Normal file
19
Tinyboard/README.md
Normal file
|
@ -0,0 +1,19 @@
|
|||
# Tinyboard
|
||||
|
||||
Makes the keyboard small and smart.
|
||||
|
||||
Few small tweaks to the keyboard:
|
||||
- Shrinks the keyboard to a size that isn't fit for grandma.
|
||||
- Adjusts keyboard placement logic to align with the menu that it spawns from.
|
||||
- Enforces a title on the keyboard input if one is not found.
|
||||
|
||||
---
|
||||
|
||||
Here is the block of text where I tell you this mod is not affiliated with or endorsed by ABI.
|
||||
https://documentation.abinteractive.net/official/legal/tos/#7-modding-our-games
|
||||
|
||||
> This mod is an independent creation not affiliated with, supported by, or approved by Alpha Blend Interactive.
|
||||
|
||||
> Use of this mod is done so at the user's own risk and the creator cannot be held responsible for any issues arising from its use.
|
||||
|
||||
> To the best of my knowledge, I have adhered to the Modding Guidelines established by Alpha Blend Interactive.
|
6
Tinyboard/Tinyboard.csproj
Normal file
6
Tinyboard/Tinyboard.csproj
Normal file
|
@ -0,0 +1,6 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
<PropertyGroup>
|
||||
<RootNamespace>YouAreMineNow</RootNamespace>
|
||||
</PropertyGroup>
|
||||
</Project>
|
23
Tinyboard/format.json
Normal file
23
Tinyboard/format.json
Normal file
|
@ -0,0 +1,23 @@
|
|||
{
|
||||
"_id": -1,
|
||||
"name": "Tinyboard",
|
||||
"modversion": "1.0.0",
|
||||
"gameversion": "2025r180",
|
||||
"loaderversion": "0.7.2",
|
||||
"modtype": "Mod",
|
||||
"author": "NotAKidoS",
|
||||
"description": "Few small tweaks to the keyboard:\n- Shrinks the keyboard to a size that isn't fit for grandma.\n- Adjusts keyboard placement logic to align with the menu that it spawns from.\n- Enforces a title on the keyboard input if one is not found.",
|
||||
"searchtags": [
|
||||
"keyboard",
|
||||
"menu",
|
||||
"ui",
|
||||
"input"
|
||||
],
|
||||
"requirements": [
|
||||
"None"
|
||||
],
|
||||
"downloadlink": "https://github.com/NotAKidoS/NAK_CVR_Mods/releases/download/r47/Tinyboard.dll",
|
||||
"sourcelink": "https://github.com/NotAKidoS/NAK_CVR_Mods/tree/main/Tinyboard/",
|
||||
"changelog": "- Initial release",
|
||||
"embedcolor": "#f61963"
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue