Files
fuckcapslock/Program.cs

192 lines
8.1 KiB
C#

// FuckCapsLock — Permanently disable the vestigial Caps Lock key.
// Copyright (c) 2026 Llama Chile Shop. MIT License.
//
// This application was pair-programmed by:
// gramps@llamachile.shop (senior developer, direct supervision)
// opencode (AI pair programmer)
//
// Under direct, senior supervision, paired programming can be an
// effective and efficient tool.
//
// Installs a low-level keyboard hook (WH_KEYBOARD_LL) that swallows
// VK_CAPITAL (0x14) before it reaches any window. Runs in the system
// tray until the user exits via the context menu.
//
// System calls reference:
// SetWindowsHookEx / UnhookWindowsHookEx / CallNextHookEx — user32.dll
// GetModuleHandle — kernel32.dll
// Registry.CurrentUser\...\Run — auto-start on boot
using System.Diagnostics;
using System.Runtime.InteropServices;
using Microsoft.Win32;
static class Program
{
/// <summary>Low-level keyboard hook identifier — intercepts keys before any window.</summary>
const int WH_KEYBOARD_LL = 13;
/// <summary>Virtual-key code for Caps Lock — the only key this app blocks.</summary>
const int VK_CAPITAL = 0x14;
/// <summary>Callback signature for LowLevelKeyboardProc used by SetWindowsHookEx.</summary>
/// <param name="nCode">Hook code — >= 0 means the hook should process the message.</param>
/// <param name="wParam">Message type (WM_KEYDOWN, WM_KEYUP, etc.).</param>
/// <param name="lParam">Pointer to a KBDLLHOOKSTRUCT with the raw key data.</param>
/// <returns>1 to swallow the key, or CallNextHookEx to let it through.</returns>
delegate IntPtr LowLevelKeyboardProc(int nCode, IntPtr wParam, IntPtr lParam);
/// <summary>Installs a low-level keyboard hook into the global hook chain.</summary>
/// <param name="idHook">Hook type — WH_KEYBOARD_LL (13) for keyboard events.</param>
/// <param name="lpfn">Pointer to the callback function.</param>
/// <param name="hMod">Module handle for the current process.</param>
/// <param name="dwThreadId">0 = hook applies globally to all threads in this desktop.</param>
/// <returns>Hook handle on success, IntPtr.Zero on failure.</returns>
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
static extern IntPtr SetWindowsHookEx(int idHook, LowLevelKeyboardProc lpfn, IntPtr hMod, uint dwThreadId);
/// <summary>Removes a previously installed hook from the hook chain.</summary>
/// <param name="hhk">Hook handle returned by SetWindowsHookEx.</param>
/// <returns>True on success.</returns>
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
[return: MarshalAs(UnmanagedType.Bool)]
static extern bool UnhookWindowsHookEx(IntPtr hhk);
/// <summary>Passes the hook event to the next hook in the chain.</summary>
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
static extern IntPtr CallNextHookEx(IntPtr hhk, int nCode, IntPtr wParam, IntPtr lParam);
/// <summary>Retrieves the module handle for the calling process — required by SetWindowsHookEx.</summary>
[DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
static extern IntPtr GetModuleHandle(string lpModuleName);
/// <summary>Structure deserialized from the lParam pointer in the hook callback.</summary>
[StructLayout(LayoutKind.Sequential)]
struct KBDLLHOOKSTRUCT
{
public uint vkCode;
public uint scanCode;
public uint flags;
public uint time;
public IntPtr dwExtraInfo;
}
/// <summary>Cached delegate — prevents garbage collection of the callback.</summary>
static LowLevelKeyboardProc _proc = HookCallback;
/// <summary>Active hook handle returned by SetWindowsHookEx.</summary>
static IntPtr _hookId = IntPtr.Zero;
/// <summary>System tray icon — visible while the app is running.</summary>
static NotifyIcon? _trayIcon;
/// <summary>
/// Entry point. Registers auto-start, enforces a single instance via a
/// named mutex, installs the keyboard hook, and runs the WinForms message
/// loop. The app lives in the system tray until Exit is selected.
/// </summary>
[STAThread]
static void Main()
{
RegisterStartup();
// Named mutex prevents multiple instances — second launch exits silently.
using var mutex = new Mutex(true, @"Global\FuckCapsLock", out var createdNew);
if (!createdNew)
return;
// Install the low-level hook; bail if it fails (e.g., insufficient permissions).
_hookId = SetHook(_proc);
if (_hookId == IntPtr.Zero)
return;
ApplicationConfiguration.Initialize();
// Hidden window — required to pump messages and keep the hook alive.
using var form = new Form();
form.WindowState = FormWindowState.Minimized;
form.ShowInTaskbar = false;
form.Load += (_, _) => form.Hide();
// Tray icon with Exit context menu to cleanly unhook and quit.
_trayIcon = new NotifyIcon
{
Icon = SystemIcons.Application,
Text = "FuckCapsLock",
ContextMenuStrip = new ContextMenuStrip()
};
_trayIcon.ContextMenuStrip.Items.Add("Exit", null, (_, _) =>
{
_trayIcon.Visible = false;
Application.Exit();
});
_trayIcon.Visible = true;
// Blocks until the form closes (Exit is clicked or process is killed).
Application.Run(form);
// Remove the hook when the app exits.
UnhookWindowsHookEx(_hookId);
}
/// <summary>
/// Writes the current executable path to HKCU\Software\Microsoft\Windows\
/// CurrentVersion\Run so the app auto-starts on login. Silently ignores
/// errors (no admin rights needed for HKCU).
/// </summary>
static void RegisterStartup()
{
try
{
using var key = Registry.CurrentUser.OpenSubKey(
@"Software\Microsoft\Windows\CurrentVersion\Run", true);
if (key is null) return;
var path = Environment.ProcessPath;
if (path is null) return;
// Quote the path if it contains spaces (standard Windows practice).
var value = path.Contains(' ') ? $"\"{path}\"" : path;
// Only write if the path has changed — avoids unnecessary registry churn.
var existing = key.GetValue("FuckCapsLock") as string;
if (!string.Equals(existing, value, StringComparison.OrdinalIgnoreCase))
key.SetValue("FuckCapsLock", value);
}
catch { }
}
/// <summary>
/// Installs the low-level keyboard hook using SetWindowsHookEx.
/// </summary>
/// <param name="proc">Delegate to the callback that processes each key event.</param>
/// <returns>Hook handle or IntPtr.Zero if installation failed.</returns>
static IntPtr SetHook(LowLevelKeyboardProc proc)
{
using var curProcess = Process.GetCurrentProcess();
using var curModule = curProcess.MainModule;
return SetWindowsHookEx(WH_KEYBOARD_LL, proc,
GetModuleHandle(curModule!.ModuleName), 0);
}
/// <summary>
/// Hook callback invoked for every keyboard event. If the key is VK_CAPITAL (Caps Lock),
/// the event is swallowed by returning 1 without calling CallNextHookEx. All other keys
/// pass through normally.
/// </summary>
/// <param name="nCode">>= 0 means the hook should process the event.</param>
/// <param name="wParam">Window message type (WM_KEYDOWN, WM_SYSKEYUP, etc.).</param>
/// <param name="lParam">Pointer to a KBDLLHOOKSTRUCT with the raw key data.</param>
/// <returns>1 if Caps Lock was blocked; otherwise the result of CallNextHookEx.</returns>
static IntPtr HookCallback(int nCode, IntPtr wParam, IntPtr lParam)
{
if (nCode >= 0)
{
var hookStruct = Marshal.PtrToStructure<KBDLLHOOKSTRUCT>(lParam);
if (hookStruct.vkCode == VK_CAPITAL)
return (IntPtr)1; // Swallow the key — never forward it.
}
return CallNextHookEx(_hookId, nCode, wParam, lParam);
}
}