Compare commits

...

22 Commits

Author SHA1 Message Date
francop 5af469ae3e Updated layout 2026-05-31 22:05:42 -04:00
francop d0e69a03fa Updates to layouts 2026-05-24 12:59:01 -04:00
francop e7a9ccde93 Establish sequential window targeting intervals 2026-05-24 12:58:44 -04:00
francop e57eb09567 Added one line inside the wake block to capture the time of the wake event: obj.wakeTimestamp = os.time() 2026-05-20 21:44:59 -04:00
francop 734b291dd6 Update to Layout 2026-05-19 11:55:40 -04:00
francop a710b6c5de Added Detection if last screen count was 1, no window rescue needed.
Added: On rescuce, start at 0,0
2026-05-19 11:55:20 -04:00
francop 75200ff664 Updated to reflect module changes 2026-05-16 22:42:29 -04:00
francop 73e5617176 Moved to modules 2026-05-16 22:42:02 -04:00
francop c9d516c6fd Added: Captured Windows Inside Profile State File 2026-05-16 20:53:06 -04:00
francop efadc1e218 Added comment: Trigger for Focusdim 2026-05-16 20:42:12 -04:00
francop 07226bfa7b Added hs.allowAppleScript(true) to allow Apple script 2026-05-16 12:18:24 -04:00
francop 21580ae0d6 Added: require("modules.menuHub").start() 2026-05-16 11:35:52 -04:00
francop 86f8072d1b Fix: Stop tracking .DS_Store files 2026-05-16 11:18:53 -04:00
francop 3c9c03b64b New files 2026-05-16 11:11:49 -04:00
francop 11ed9b80bc local SPEED_TO_DISABLE = 140.0 2026-05-16 11:10:43 -04:00
francop c3308429ff The Menu Render Guard implemetned to prevent IPC errors after sleep/wake 2026-05-16 11:10:16 -04:00
francop 4b1d6808c9 Added MouseJiggle Module 2026-05-16 03:18:30 -04:00
francop 940ed913aa Added: MouseJiggle 2026-05-16 02:55:15 -04:00
francop 1990c2f998 Removed focusmode 2026-05-16 02:00:29 -04:00
francop e16532cf1b Added Focusdim as ignore app 2026-05-16 02:00:09 -04:00
francop 6a0a954e7e Initial Version 2026-05-16 01:21:57 -04:00
francop d4ff06d1a3 Added require("FocusMode") 2026-05-16 01:21:39 -04:00
17 changed files with 445 additions and 158 deletions
Vendored
BIN
View File
Binary file not shown.
+38
View File
@@ -0,0 +1,38 @@
-- FocusMode.lua
local obj = {}
obj.isActive = false
-- Load the native C-optimized window highlighting extension
local hl = require("hs.window.highlight")
-- Configure the native isolation overlay properties
-- 78% black mask for deep workspace distraction isolation
hl.ui.isolateColor = {0, 0, 0, 0.78}
-- Start the background layout tracking engine loop natively
hl.start()
function obj.toggle()
-- Native switch to toggle isolate highlight state instantly
hl.toggleIsolate()
obj.isActive = not obj.isActive
if obj.isActive then
hs.alert.show("Focus Mode Active", 1.5)
else
hs.alert.show("Focus Mode Disabled", 1.5)
end
end
-- Native Hotkey Binding (Maps to your global hyper + F layout configuration)
if hyper then
hs.hotkey.bind(hyper, "F", function()
obj.toggle()
end)
end
-- Export module instance globally for our external Raycast script runner triggers
FocusMode = obj
return obj
BIN
View File
Binary file not shown.
+38 -3
View File
@@ -166,9 +166,14 @@ local function executeRestore(filePath, layoutName)
win:setFrame({x=x, y=y, w=w, h=h}, 0)
end
end
-- Establish sequential window targeting intervals
-- Shifts execution past the window manager canvas updates
moveAction()
hs.timer.doAfter(0.5, moveAction)
hs.timer.doAfter(1.5, moveAction)
local intervals = isStubborn and {0.2, 0.6, 1.2, 2.2} or {0.3, 1.0}
for _, delay in ipairs(intervals) do
hs.timer.doAfter(delay, moveAction)
end
end
end
end
@@ -178,7 +183,7 @@ local function executeRestore(filePath, layoutName)
if launchedAny then
hs.alert.show("Syncing Apps...", 3)
hs.timer.doAfter(4.5, moveWindows)
hs.timer.doAfter(5.0, moveWindows) -- Bumped to 5s to let heavy frameworks build window handles completely
else
moveWindows()
end
@@ -235,9 +240,39 @@ function selector.showLayoutActions(layoutName)
text = "🗑️ Delete Layout Profile",
subText = "Permanently remove the profile config file from disk",
action = "delete"
},
{
text = "───────────────────────────────────────────────────────",
subText = "📦 Captured Windows Inside Profile State File:",
action = "info"
}
}
-- Dynamically read layout profile file and append saved windows
local data = hs.json.read(path)
if data and data.windows and #data.windows > 0 then
for _, win in ipairs(data.windows) do
-- Verify if the application is currently running on the system server
local isRunning = hs.application.get(win.bundleID) or hs.application.get(win.appName)
local statusIndicator = isRunning and "🟢" or "🔴"
local cleanTitle = (win.winTitle and win.winTitle ~= "") and win.winTitle or "Untitled Window"
if #cleanTitle > 60 then cleanTitle = string.sub(cleanTitle, 1, 57) .. "..." end
table.insert(actionChoices, {
text = string.format("%s %s", statusIndicator, win.appName),
subText = string.format("↳ Title: \"%s\" | Bounds: %dx%d at (%d,%d)", cleanTitle, win.w, win.h, win.x, win.y),
action = "info"
})
end
else
table.insert(actionChoices, {
text = "⚠️ No Saved Window Metrics Found",
subText = "This layout profile does not contain any captured windows.",
action = "info"
})
end
if selector.actionChooser then selector.actionChooser:hide() end
selector.actionChooser = hs.chooser.new(function(choice)
BIN
View File
Binary file not shown.
+58 -7
View File
@@ -34,9 +34,11 @@ obj.saveCountdown = obj.saveInterval
obj.isRescued = false
obj.isTransitioning = false
obj.isRestoring = false
obj.isMenuUpdating = false -- Safety guard to prevent concurrent IPC menu rendering loops
obj.wakeTimer = nil
obj.lastScreenCount = #hs.screen.allScreens()
obj.lastSavedTime = "Never"
obj.wakeTimestamp = 0 -- Track wake time to isolate screen changes during system startup
-- ==========================================
-- INTERNAL UTILITIES
@@ -210,8 +212,8 @@ function obj.rescueWindowsToLaptop()
if f.w > maxFrame.w then f.w = maxFrame.w - 100 end
if f.h > maxFrame.h then f.h = maxFrame.h - 100 end
f.x = maxFrame.x + 50 + staggerOffset
f.y = maxFrame.y + 50 + staggerOffset
f.x = maxFrame.x + staggerOffset
f.y = maxFrame.y + staggerOffset
win:setFrame(f, 0)
staggerOffset = staggerOffset + 30
@@ -231,6 +233,9 @@ end
local timerMenu = hs.menubar.new()
function updateMenu()
if obj.isMenuUpdating then return end
obj.isMenuUpdating = true
if timerMenu then
local screens = hs.screen.allScreens()
timerMenu:setTitle(string.format("💠 %d:%02d", math.floor(obj.saveCountdown / 60), obj.saveCountdown % 60))
@@ -257,6 +262,8 @@ function updateMenu()
end
timerMenu:setMenu(menuTable)
end
obj.isMenuUpdating = false
end
obj.powerWatcher = hs.caffeinate.watcher.new(function(event)
@@ -264,27 +271,39 @@ obj.powerWatcher = hs.caffeinate.watcher.new(function(event)
log("POWER: Sleep event.")
obj.isTransitioning = true
if obj.autoSaveTimer then obj.autoSaveTimer:stop() end
if obj.clockTimer then obj.clockTimer:stop() end
elseif event == hs.caffeinate.watcher.systemDidWake or event == hs.caffeinate.watcher.screensDidWake or event == hs.caffeinate.watcher.screensDidUnlock then
log("POWER: Wake event.")
obj.saveCountdown = obj.saveInterval
obj.isMenuUpdating = false
obj.isTransitioning = false
obj.isRestoring = false
obj.wakeTimestamp = os.time()
if obj.autoSaveTimer then obj.autoSaveTimer:start() end
if obj.clockTimer then obj.clockTimer:start() end
if obj.wakeTimer then obj.wakeTimer:stop() end
local currentScreens = #hs.screen.allScreens()
if currentScreens > 1 then
log("WAKE ACTIVATE: Multi-screen detected on wake. Scheduling layout restoration.")
obj.lastScreenCount = currentScreens -- Sync variable immediately so it knows we are docked
obj.wakeTimer = hs.timer.doAfter(12, function()
obj.isTransitioning = false
obj.isRestoring = false
obj.lastScreenCount = currentScreens
obj.restoreLayout()
obj.wakeTimer = nil
end)
else
log("WAKE SKIP: Single screen detected. Syncing count only.")
obj.isTransitioning = false
obj.isRestoring = false
if obj.lastScreenCount > 1 then
log("WAKE ACTIVATE: Screen count dropped from " .. obj.lastScreenCount .. " to 1 during sleep. Running rescue.")
obj.rescueWindowsToLaptop()
else
log("WAKE SKIP: Woke up on single screen, matched previous state. No rescue needed.")
end
obj.lastScreenCount = currentScreens
updateMenu()
end
end
end):start()
@@ -294,6 +313,11 @@ hs.hotkey.bind({"shift", "cmd"}, "R", function() obj.restoreLayout() end)
hs.hotkey.bind({"shift", "cmd", "ctrl"}, "L", obj.rescueWindowsToLaptop)
obj.screenWatcher = hs.screen.watcher.new(function()
if (os.time() - obj.wakeTimestamp) < 10 then
log("DOCK EVENT: Dropped via wake isolation guard.")
return
end
if obj.isTransitioning or obj.isRestoring or obj.wakeTimer then
log("DOCK EVENT: Ignored.")
return
@@ -364,9 +388,37 @@ function obj.showMenu()
text = "🚀 Rescue Windows",
subText = "Cascade active window threads onto primary laptop screen space",
action = "rescue"
},
{
text = "───────────────────────────────────────────────────────",
subText = "📦 Captured Windows Inside Profile State File:",
action = "info"
}
}
local data = hs.json.read(obj.layoutFile)
if data and data.windows and #data.windows > 0 then
for _, win in ipairs(data.windows) do
local isRunning = hs.application.get(win.bundleID) or hs.application.get(win.appName)
local statusIndicator = isRunning and "🟢" or "🔴"
local cleanTitle = (win.winTitle and win.winTitle ~= "") and win.winTitle or "Untitled Window"
if #cleanTitle > 60 then cleanTitle = string.sub(cleanTitle, 1, 57) .. "..." end
table.insert(choices, {
text = string.format("%s %s", statusIndicator, win.appName),
subText = string.format("↳ Title: \"%s\" | Bounds: %dx%d at (%d,%d)", cleanTitle, win.w, win.h, win.x, win.y),
action = "info"
})
end
else
table.insert(choices, {
text = "⚠️ No Saved Window Metrics Found",
subText = "Run a Save operation to record your desktop layout profile context.",
action = "info"
})
end
if obj.instanceChooser then obj.instanceChooser:hide() end
obj.instanceChooser = hs.chooser.new(function(choice)
@@ -387,7 +439,6 @@ function obj.showMenu()
obj.instanceChooser:show()
end
-- Export module instance globally for direct IPC command routing
WindowManager = obj
return obj
+2 -1
View File
@@ -28,6 +28,7 @@
"TypeWhisper": true,
"com.typewhisper.mac": true,
"dk.heyiam.monocle": true,
"Monocle": true
"Monocle": true,
"com.privdev.Focusdim": true
}
}
+17
View File
@@ -0,0 +1,17 @@
{
"Work": {
"keepOpen": ["Slack", "Microsoft Teams", "Visual Studio Code", "Terminal", "Google Chrome"],
"close": ["Discord", "Steam", "Komga", "Kavita"],
"dimAlpha": 0.3
},
"Home": {
"keepOpen": ["Code", "Spark", "Spotify"],
"close": ["Slack", "Microsoft Teams"],
"dimAlpha": 0.7
},
"Study": {
"keepOpen": ["AFFiNE", "Obsidian", "Anki", "Preview", "Safari"],
"close": ["Slack", "Microsoft Teams", "Discord", "Steam", "Instagram"],
"dimAlpha": 0.5
}
}
+21 -72
View File
@@ -1,104 +1,53 @@
-- ========================================================================
-- HEADLESS WORKSPACE BACKGROUND ENGINE (FORCED AT BOOT)
-- ========================================================================
hs.allowAppleScript(true)
hs.menuIcon(false)
hs.dockIcon(false)
hs.ipc.cliInstall("/opt/homebrew")
hs.alert.show("Hammerspoon Headless Daemon Active", 2)
-- ~/.hammerspoon/init.lua
-- Load Config First
-- config = require("Config")
--
-- Load your HyperKey
-- Core Subsystems and Global Bindings
require("HyperKey")
require('SearchWindows')
require('Caffeine')
require('AppBorders')
-- CRITICAL FOR RAYCAST INTERACTION: Bind the return value to a global variable
-- IPC Bridges & Trackers
LayoutSelector = require('LayoutSelector')
require('System_Tweaks') -- Used for Time Machine Throttle Disable
-- require("Focus") -- Does not work with layout saver - Not needed if using Monocle
Network = require("NetworkCenter")
-- Load the window management module
-- Active Modules
require("modules.mouseJiggle").start()
local windowMgr = require("WindowManager")
local productivity = require("productivity")
-- For Affine
-- DEEP FOCUS MODULE (New Integration)
local focus = require("modules.focus")
-- Affine Note Engine Integration
local quickNote = require("affine_quick_note")
quickNote.init()
require("affine_clipper"):init()
-- Load Spoon Files
-- Spoons Engine & External Tool Integrations
hs.loadSpoon('SpoonInstall')
hs.loadSpoon('SpeedMenu')
hs.loadSpoon('BrewInfo')
-- ==========================================
-- SPOON CONFIGURATION
-- ==========================================
-- Run isolated Spoon setup configurations
require("modules.spoon_config")
---- SpeedMenu Config
if spoon.SpeedMenu then
-- Define the Fix Function (includes MAC and IPv6)
local function applyFullMenuFix()
local interface = spoon.SpeedMenu.interface or "en0"
local ssid = hs.wifi.currentNetwork() or "Disconnected"
local details = hs.network.interfaceDetails(interface)
local ipv4 = (details and details.IPv4) and details.IPv4.Addresses[1] or "N/A"
local ipv6 = (details and details.IPv6) and details.IPv6.Addresses[1] or "N/A"
-- Get MAC Address via shell
local macaddr = hs.execute('ifconfig ' .. interface .. ' | grep ether | awk \'{print $2}\''):gsub("%s+", "")
local menuitems = {
{ title = "SSID: " .. ssid, fn = function() hs.pasteboard.setContents(ssid) end },
{ title = "IPv4: " .. ipv4, fn = function() hs.pasteboard.setContents(ipv4) end },
{ title = "IPv6: " .. ipv6, fn = function() hs.pasteboard.setContents(ipv6) end },
{ title = "MAC: " .. macaddr, fn = function() hs.pasteboard.setContents(macaddr) end },
{ title = "-" },
{ title = "Rescan Network Interfaces", fn = function() spoon.SpeedMenu:rescan() end }
}
spoon.SpeedMenu.menubar:setMenu(menuitems)
end
-- Hook the rescan method
local oldRescan = spoon.SpeedMenu.rescan
spoon.SpeedMenu.rescan = function(self)
oldRescan(self)
applyFullMenuFix()
end
-- Toggle Logic (Starts as OFF)
local speedMenuRunning = false
hs.hotkey.bind({"cmd", "alt", "ctrl"}, "W", function()
if speedMenuRunning then
spoon.SpeedMenu:stop()
hs.alert.show("SpeedMenu Stopped")
else
spoon.SpeedMenu:start() -- This puts it in the menu bar
applyFullMenuFix() -- This populates the data
hs.alert.show("SpeedMenu Started")
end
speedMenuRunning = not speedMenuRunning
end)
-- ========================================================================
-- RAYCAST / INTER-PROCESS COMMUNICATION (IPC) BRIDGE BINDINGS
-- ========================================================================
-- Expose the focus module methods globally so the hs CLI tool can call them
function ActivateDeepFocus(profileName)
return focus.activate(profileName)
end
---- SpeedMenu Config END
---- BrewInfo
if spoon.BrewInfo then
hs.hotkey.bind({"cmd", "alt", "ctrl"}, "I", function()
spoon.BrewInfo:showBrewInfo()
end)
hs.hotkey.bind({"cmd", "alt", "ctrl"}, "J", function()
spoon.BrewInfo:showBrewInfoCurSel()
end)
function ClearDeepFocus()
return focus.clear()
end
---- BrewInfo END
-- Boot Confirmation
hs.alert.show("Hammerspoon Config Reloaded")
BIN
View File
Binary file not shown.
+14 -14
View File
@@ -3,50 +3,50 @@
{
"x" : 1920,
"winTitle" : "AFFiNE",
"y" : 0,
"appName" : "AFFiNE",
"y" : 0,
"h" : 1080,
"w" : 960,
"bundleID" : "pro.affine.app"
},
{
"x" : 0,
"winTitle" : "TrueNAS - 192.168.1.135 - Google Chrome",
"winTitle" : "[D] Coding Workspace.json (Working Tree) ([D] Coding Workspace.json) — .hammerspoon",
"appName" : "Code",
"y" : 30,
"appName" : "Google Chrome",
"h" : 957,
"h" : 956,
"w" : 1920,
"bundleID" : "com.google.Chrome"
"bundleID" : "com.microsoft.VSCode"
},
{
"x" : 2880,
"winTitle" : "francop — -zsh — 134×33",
"y" : 536,
"winTitle" : "francop — francop@media: \/mnt\/media\/Movies — -zsh — 134×33",
"appName" : "Terminal",
"y" : 536,
"h" : 544,
"w" : 958,
"bundleID" : "com.apple.Terminal"
},
{
"x" : 2880,
"winTitle" : "francop — tail -f \/tmp\/litellm.out.log — 134×33",
"y" : 0,
"winTitle" : "raycast-scripts — francop@nextcloud: \/mnt\/docs\/Consume — -zsh — 134×33",
"appName" : "Terminal",
"y" : 0,
"h" : 544,
"w" : 958,
"bundleID" : "com.apple.Terminal"
},
{
"x" : 0,
"winTitle" : "WindowManager.lua — .hammerspoon",
"winTitle" : "anker desk outlet - Google Search - Google Chrome",
"appName" : "Google Chrome",
"y" : 30,
"appName" : "Code",
"h" : 956,
"h" : 957,
"w" : 1920,
"bundleID" : "com.microsoft.VSCode"
"bundleID" : "com.google.Chrome"
}
],
"screenCount" : 3,
"saveTime" : "2026-05-08 23:19:07",
"saveTime" : "2026-05-24 14:34:05",
"mode" : "Docked"
}
-61
View File
@@ -1,61 +0,0 @@
{
"screenCount" : 3,
"windows" : [
{
"x" : -952,
"bundleID" : "com.apple.finder",
"y" : 159,
"appName" : "Finder",
"h" : 492,
"w" : 920,
"winTitle" : "raycast-scripts"
},
{
"x" : 2070,
"bundleID" : "com.raycast.macos",
"y" : 121,
"appName" : "Raycast",
"h" : 319,
"w" : 1000,
"winTitle" : "Settings"
},
{
"x" : 1986,
"bundleID" : "com.google.GeminiMacOS",
"y" : 110,
"appName" : "Gemini",
"h" : 536,
"w" : 1002,
"winTitle" : "Gemini — Hammerspoon Menu Bar Icon Troubleshooting"
},
{
"x" : 0,
"bundleID" : "com.google.Chrome",
"y" : 39,
"appName" : "Google Chrome",
"h" : 957,
"w" : 1920,
"winTitle" : "francop - Dashboard - Gitea: Git with a cup of tea - Google Chrome"
},
{
"x" : 2794,
"bundleID" : "com.apple.Terminal",
"y" : 131,
"appName" : "Terminal",
"h" : 499,
"w" : 860,
"winTitle" : "francop — -zsh — 120×30"
},
{
"x" : 0,
"bundleID" : "com.microsoft.VSCode",
"y" : 30,
"appName" : "Code",
"h" : 957,
"w" : 1920,
"winTitle" : "init.lua — .hammerspoon"
}
],
"saveTime" : "2026-05-15 22:26:48",
"mode" : "Docked"
}
+10
View File
@@ -0,0 +1,10 @@
local focus = require("modules.focus")
-- Expose clean functions globally for the hs CLI tool
function ActivateDeepFocus(profileName)
return focus.activate(profileName)
end
function ClearDeepFocus()
return focus.clear()
end
+88
View File
@@ -0,0 +1,88 @@
local M = {}
local focusCanvas = nil
local activeProfile = nil
local configPath = hs.configdir .. "/focus_profiles.json"
local function loadProfiles()
local file = io.open(configPath, "r")
if not file then
print("Error: Could not open focus_profiles.json")
return nil
end
local content = file:read("*a")
file:close()
return hs.json.decode(content)
end
local function manageApps(profileConfig)
for _, appName in ipairs(profileConfig.close) do
local app = hs.application.get(appName)
if app then app:kill() end
end
for _, appName in ipairs(profileConfig.keepOpen) do
hs.application.launchOrFocus(appName)
end
end
local function setFocusOverlay(alpha)
if focusCanvas then
focusCanvas:delete()
focusCanvas = nil
end
if not alpha or alpha == 0 then return end
local mainScreen = hs.screen.mainScreen()
local rect = mainScreen:frame()
focusCanvas = hs.canvas.new(rect)
focusCanvas:insertElement({
action = "fill",
type = "rectangle",
fillColor = { red = 0, green = 0, blue = 0, alpha = alpha }
})
focusCanvas:level(50)
focusCanvas:show()
local frontApp = hs.application.frontmostApplication()
if frontApp then
local win = frontApp:mainWindow()
if win then win:focus() end
end
end
function M.activate(profileName)
local profiles = loadProfiles()
if not profiles or not profiles[profileName] then
print("Error: Profile '" .. tostring(profileName) .. "' not found.")
return "Profile not found"
end
activeProfile = profileName
local currentProfileConfig = profiles[profileName]
manageApps(currentProfileConfig)
setFocusOverlay(currentProfileConfig.dimAlpha)
return "Activated " .. profileName .. " mode"
end
function M.clear()
setFocusOverlay(nil)
activeProfile = nil
return "Focus mode cleared"
end
hs.screen.watcher.new(function()
if activeProfile then
local profiles = loadProfiles()
if profiles and profiles[activeProfile] then
setFocusOverlay(profiles[activeProfile].dimAlpha)
end
end
end):start()
return M
+99
View File
@@ -0,0 +1,99 @@
local M = {}
-- Configuration
local REQUIRED_REVERSALS = 3 -- Must switch direction 3 times (e.g., Left -> Right -> Left)
local RESET_TIMEOUT = 0.3 -- Fast window: shake must be completed rapidly or it resets
local LOCKOUT_DURATION = 3.0 -- Cooldown duration (seconds) before allowed to trigger again
-- Asymmetrical Thresholds
local SPEED_TO_ENABLE = 115.0 -- Slightly lower: easier to turn ON
local SPEED_TO_DISABLE = 140.0 -- Slightly higher: harder to accidentally turn OFF
-- Focusdim Global Hotkey Configuration from your settings
local FOCUS_DIM_MODS = { "ctrl", "alt", "shift", "cmd" }
local FOCUS_DIM_KEY = "f"
-- Internal state tracking
local jiggleWatcher = nil
local isLocked = false
local isFocusDimActive = false -- Tracks state to apply the correct threshold
local lastXDirection = 0 -- -1 for left, 1 for right, 0 for still
local reversalCount = 0
local lastEventTime = 0
-- Forward declaration for the timer loop
M.start = function() end
local function eventCallback(event)
if isLocked then return false end
local now = hs.timer.secondsSinceEpoch()
-- Reset the counter if the movements aren't happening fast enough
if (now - lastEventTime) > RESET_TIMEOUT then
reversalCount = 0
lastXDirection = 0
end
lastEventTime = now
-- Extract horizontal movement exclusively
local dx = event:getProperty(hs.eventtap.event.properties.mouseEventDeltaX)
-- Dynamically choose threshold based on current state
local currentThreshold = isFocusDimActive and SPEED_TO_DISABLE or SPEED_TO_ENABLE
-- Only evaluate if it's a high-velocity horizontal snap matching the current state's rule
if math.abs(dx) > currentThreshold then
local currentDirection = (dx > 0) and 1 or -1
-- Track direction flips
if lastXDirection ~= 0 and currentDirection ~= lastXDirection then
reversalCount = reversalCount + 1
-- Trigger only when the full rapid shake sequence completes
if reversalCount >= REQUIRED_REVERSALS then
isLocked = true
reversalCount = 0
lastXDirection = 0
-- Toggle our internal tracking state
isFocusDimActive = not isFocusDimActive
-- Instantly detach the listener
M.stop()
-- Simulate the hotkey press to toggle Focusdim
hs.eventtap.keyStroke(FOCUS_DIM_MODS, FOCUS_DIM_KEY, 0)
-- Re-enable after the lockout duration
hs.timer.doAfter(LOCKOUT_DURATION, function()
isLocked = false
M.start()
end)
return false
end
end
lastXDirection = currentDirection
end
return false -- Pass event through to macOS normally
end
function M.start()
if jiggleWatcher then return end
jiggleWatcher = hs.eventtap.new({ hs.eventtap.event.types.mouseMoved }, eventCallback)
jiggleWatcher:start()
end
function M.stop()
if jiggleWatcher then
jiggleWatcher:stop()
jiggleWatcher = nil
end
end
return M
+60
View File
@@ -0,0 +1,60 @@
-- ========================================================================
-- SPOON RUNTIME CONFIGURATION MODULE
-- ========================================================================
---- SpeedMenu Implementation
if spoon.SpeedMenu then
-- Define the Fix Function (includes MAC and IPv6 parsing via shell)
local function applyFullMenuFix()
local interface = spoon.SpeedMenu.interface or "en0"
local ssid = hs.wifi.currentNetwork() or "Disconnected"
local details = hs.network.interfaceDetails(interface)
local ipv4 = (details and details.IPv4) and details.IPv4.Addresses[1] or "N/A"
local ipv6 = (details and details.IPv6) and details.IPv6.Addresses[1] or "N/A"
-- Get MAC Address via native shell pipe
local macaddr = hs.execute('ifconfig ' .. interface .. ' | grep ether | awk \'{print $2}\''):gsub("%s+", "")
local menuitems = {
{ title = "SSID: " .. ssid, fn = function() hs.pasteboard.setContents(ssid) end },
{ title = "IPv4: " .. ipv4, fn = function() hs.pasteboard.setContents(ipv4) end },
{ title = "IPv6: " .. ipv6, fn = function() hs.pasteboard.setContents(ipv6) end },
{ title = "MAC: " .. macaddr, fn = function() hs.pasteboard.setContents(macaddr) end },
{ title = "-" },
{ title = "Rescan Network Interfaces", fn = function() spoon.SpeedMenu:rescan() end }
}
spoon.SpeedMenu.menubar:setMenu(menuitems)
end
-- Overwrite the baseline rescan hook cleanly
local oldRescan = spoon.SpeedMenu.rescan
spoon.SpeedMenu.rescan = function(self)
oldRescan(self)
applyFullMenuFix()
end
-- Toggle Engine State Manager
local speedMenuRunning = false
hs.hotkey.bind({"cmd", "alt", "ctrl"}, "W", function()
if speedMenuRunning then
spoon.SpeedMenu:stop()
hs.alert.show("SpeedMenu Stopped")
else
spoon.SpeedMenu:start()
applyFullMenuFix()
hs.alert.show("SpeedMenu Started")
end
speedMenuRunning = not speedMenuRunning
end)
end
---- BrewInfo Implementation
if spoon.BrewInfo then
hs.hotkey.bind({"cmd", "alt", "ctrl"}, "I", function()
spoon.BrewInfo:showBrewInfo()
end)
hs.hotkey.bind({"cmd", "alt", "ctrl"}, "J", function()
spoon.BrewInfo:showBrewInfoCurSel()
end)
end
BIN
View File
Binary file not shown.