dotfiles/talon/community/community-cursorless-0.4.0/mouse_grid/mouse_grid.py
2024-11-16 20:27:38 -07:00

295 lines
9.8 KiB
Python

# courtesy of https://github.com/timo/
# see https://github.com/timo/talon_scripts
from talon import Module, Context, app, canvas, screen, settings, ui, ctrl, cron
from talon.skia import Shader, Color, Paint, Rect
from talon.types.point import Point2d
from talon_plugins import eye_mouse, eye_zoom_mouse
from typing import Union
import math, time
import typing
mod = Module()
narrow_expansion = mod.setting(
"grid_narrow_expansion",
type=int,
default=0,
desc="""After narrowing, grow the new region by this many pixels in every direction, to make things immediately on edges easier to hit, and when the grid is at its smallest, it allows you to still nudge it around""",
)
mod.tag("mouse_grid_showing", desc="Tag indicates whether the mouse grid is showing")
mod.tag("mouse_grid_enabled", desc="Tag enables the mouse grid commands.")
ctx = Context()
class MouseSnapNine:
def __init__(self):
self.screen = None
self.rect = None
self.history = []
self.img = None
self.mcanvas = None
self.active = False
self.count = 0
self.was_control_mouse_active = False
self.was_zoom_mouse_active = False
def setup(self, *, rect: Rect = None, screen_num: int = None):
screens = ui.screens()
# each if block here might set the rect to None to indicate failure
if rect is not None:
try:
screen = ui.screen_containing(*rect.center)
except Exception:
rect = None
if rect is None and screen_num is not None:
screen = screens[screen_num % len(screens)]
rect = screen.rect
if rect is None:
screen = screens[0]
rect = screen.rect
self.rect = rect.copy()
self.screen = screen
self.count = 0
self.img = None
if self.mcanvas is not None:
self.mcanvas.close()
self.mcanvas = canvas.Canvas.from_screen(screen)
if self.active:
self.mcanvas.register("draw", self.draw)
self.mcanvas.freeze()
def show(self):
if self.active:
return
# noinspection PyUnresolvedReferences
if eye_zoom_mouse.zoom_mouse.enabled:
self.was_zoom_mouse_active = True
eye_zoom_mouse.toggle_zoom_mouse(False)
if eye_mouse.control_mouse.enabled:
self.was_control_mouse_active = True
eye_mouse.control_mouse.toggle()
self.mcanvas.register("draw", self.draw)
self.mcanvas.freeze()
self.active = True
return
def close(self):
if not self.active:
return
self.mcanvas.unregister("draw", self.draw)
self.mcanvas.close()
self.mcanvas = None
self.img = None
self.active = False
if self.was_control_mouse_active and not eye_mouse.control_mouse.enabled:
eye_mouse.control_mouse.toggle()
if self.was_zoom_mouse_active and not eye_zoom_mouse.zoom_mouse.enabled:
eye_zoom_mouse.toggle_zoom_mouse(True)
self.was_zoom_mouse_active = False
self.was_control_mouse_active = False
def draw(self, canvas):
paint = canvas.paint
def draw_grid(offset_x, offset_y, width, height):
canvas.draw_line(
offset_x + width // 3,
offset_y,
offset_x + width // 3,
offset_y + height,
)
canvas.draw_line(
offset_x + 2 * width // 3,
offset_y,
offset_x + 2 * width // 3,
offset_y + height,
)
canvas.draw_line(
offset_x,
offset_y + height // 3,
offset_x + width,
offset_y + height // 3,
)
canvas.draw_line(
offset_x,
offset_y + 2 * height // 3,
offset_x + width,
offset_y + 2 * height // 3,
)
def draw_crosses(offset_x, offset_y, width, height):
for row in range(0, 2):
for col in range(0, 2):
cx = offset_x + width / 6 + (col + 0.5) * width / 3
cy = offset_y + height / 6 + (row + 0.5) * height / 3
canvas.draw_line(cx - 10, cy, cx + 10, cy)
canvas.draw_line(cx, cy - 10, cx, cy + 10)
grid_stroke = 1
def draw_text(offset_x, offset_y, width, height):
canvas.paint.text_align = canvas.paint.TextAlign.CENTER
for row in range(3):
for col in range(3):
text_string = ""
if settings["user.grids_put_one_bottom_left"]:
text_string = f"{(2 - row)*3+col+1}"
else:
text_string = f"{row*3+col+1}"
text_rect = canvas.paint.measure_text(text_string)[1]
background_rect = text_rect.copy()
background_rect.center = Point2d(
offset_x + width / 6 + col * width / 3,
offset_y + height / 6 + row * height / 3,
)
background_rect = background_rect.inset(-4)
paint.color = "9999995f"
paint.style = Paint.Style.FILL
canvas.draw_rect(background_rect)
paint.color = "00ff00ff"
canvas.draw_text(
text_string,
offset_x + width / 6 + col * width / 3,
offset_y + height / 6 + row * height / 3 + text_rect.height / 2,
)
if self.count < 2:
paint.color = "00ff007f"
for which in range(1, 10):
gap = 35 - self.count * 10
if not self.active:
gap = 45
draw_crosses(*self.calc_narrow(which, self.rect))
paint.stroke_width = grid_stroke
if self.active:
paint.color = "ff0000ff"
else:
paint.color = "000000ff"
if self.count >= 2:
aspect = self.rect.width / self.rect.height
if aspect >= 1:
w = self.screen.width / 3
h = w / aspect
else:
h = self.screen.height / 3
w = h * aspect
x = self.screen.x + (self.screen.width - w) / 2
y = self.screen.y + (self.screen.height - h) / 2
self.draw_zoom(canvas, x, y, w, h)
draw_grid(x, y, w, h)
draw_text(x, y, w, h)
else:
draw_grid(self.rect.x, self.rect.y, self.rect.width, self.rect.height)
paint.textsize += 12 - self.count * 3
draw_text(self.rect.x, self.rect.y, self.rect.width, self.rect.height)
def calc_narrow(self, which, rect):
rect = rect.copy()
bdr = narrow_expansion.get()
row = int(which - 1) // 3
col = int(which - 1) % 3
if settings["user.grids_put_one_bottom_left"]:
row = 2 - row
rect.x += int(col * rect.width // 3) - bdr
rect.y += int(row * rect.height // 3) - bdr
rect.width = (rect.width // 3) + bdr * 2
rect.height = (rect.height // 3) + bdr * 2
return rect
def narrow(self, which, move=True):
if which < 1 or which > 9:
return
self.save_state()
rect = self.calc_narrow(which, self.rect)
# check count so we don't bother zooming in _too_ far
if self.count < 5:
self.rect = rect.copy()
self.count += 1
if move:
ctrl.mouse_move(*rect.center)
if self.count >= 2:
self.update_screenshot()
else:
self.mcanvas.freeze()
def update_screenshot(self):
def finish_capture():
self.img = screen.capture_rect(self.rect)
self.mcanvas.freeze()
self.mcanvas.hide()
cron.after("16ms", finish_capture)
def draw_zoom(self, canvas, x, y, w, h):
if self.img:
src = Rect(0, 0, self.img.width, self.img.height)
dst = Rect(x, y, w, h)
canvas.draw_image_rect(self.img, src, dst)
def narrow_to_pos(self, x, y):
col_size = int(self.width // 3)
row_size = int(self.height // 3)
col = math.floor((x - self.rect.x) / col_size)
row = math.floor((y - self.rect.x) / row_size)
self.narrow(1 + col + 3 * row, move=False)
def save_state(self):
self.history.append((self.count, self.rect.copy()))
def go_back(self):
# FIXME: need window and screen tracking
self.count, self.rect = self.history.pop()
self.mcanvas.freeze()
mg = MouseSnapNine()
@mod.action_class
class GridActions:
def grid_activate():
"""Show mouse grid"""
if not mg.mcanvas:
mg.setup()
mg.show()
ctx.tags = ["user.mouse_grid_showing"]
def grid_place_window():
"""Places the grid on the currently active window"""
mg.setup(rect=ui.active_window().rect)
def grid_reset():
"""Resets the grid to fill the whole screen again"""
if mg.active:
mg.setup()
def grid_select_screen(screen: int):
"""Brings up mouse grid"""
mg.setup(screen_num=screen - 1)
mg.show()
def grid_narrow_list(digit_list: typing.List[str]):
"""Choose fields multiple times in a row"""
for d in digit_list:
GridActions.grid_narrow(int(d))
def grid_narrow(digit: Union[int, str]):
"""Choose a field of the grid and narrow the selection down"""
mg.narrow(int(digit))
def grid_go_back():
"""Sets the grid state back to what it was before the last command"""
mg.go_back()
def grid_close():
"""Close the active grid"""
ctx.tags = []
mg.close()