Files
kitty-mirror/kitty/client.py
2025-08-24 20:14:14 +05:30

327 lines
7.1 KiB
Python

#!/usr/bin/env python
# License: GPL v3 Copyright: 2016, Kovid Goyal <kovid at kovidgoyal.net>
# Replay the log from --dump-commands. To use first run
# kitty --dump-commands > file.txt
# then run
# kitty --replay-commands file.txt
# will replay the commands and pause at the end waiting for user to press enter
import json
import sys
from contextlib import suppress
from typing import Any
from .fast_data_types import TEXT_SIZE_CODE
CSI = '\x1b['
OSC = '\x1b]'
def write(x: str) -> None:
sys.stdout.write(x)
sys.stdout.flush()
def set_title(*args: Any) -> None:
pass
def set_icon(*args: Any) -> None:
pass
def screen_bell() -> None:
pass
def screen_normal_keypad_mode() -> None:
write('\x1b>')
def screen_alternate_keypad_mode() -> None:
write('\x1b=')
def screen_cursor_position(y: int, x: int) -> None:
write(f'{CSI}{y};{x}H')
def screen_cursor_forward(amt: int) -> None:
write(f'{CSI}{amt}C')
def screen_save_cursor() -> None:
write('\x1b7')
def screen_restore_cursor() -> None:
write('\x1b8')
def screen_cursor_back1(amt: int) -> None:
write(f'{CSI}{amt}D')
def screen_save_modes() -> None:
write(f'{CSI}?s')
def screen_restore_modes() -> None:
write(f'{CSI}?r')
def screen_designate_charset(which: int, to: int) -> None:
w = '()'[int(which)]
t = chr(int(to))
write(f'\x1b{w}{t}')
def select_graphic_rendition(payload: str) -> None:
write(f'{CSI}{payload}m')
def deccara(*a: int) -> None:
write(f'{CSI}{";".join(map(str, a))}$r')
def screen_cursor_to_column(c: int) -> None:
write(f'{CSI}{c}G')
def screen_cursor_to_line(ln: int) -> None:
write(f'{CSI}{ln}d')
def screen_set_mode(x: int, private: bool) -> None:
write(f'{CSI}{"?" if private else ""}{x}h')
def screen_save_mode(x: int, private: bool) -> None:
write(f'{CSI}{"?" if private else ""}{x}s')
def screen_reset_mode(x: int, private: bool) -> None:
write(f'{CSI}{"?" if private else ""}{x}l')
def screen_restore_mode(x: int, private: bool) -> None:
write(f'{CSI}{"?" if private else ""}{x}r')
def screen_set_margins(t: int, b: int) -> None:
write(f'{CSI}{t};{b}r')
def screen_indexn(n: int) -> None:
write(f'{CSI}{n}S')
def screen_delete_characters(count: int) -> None:
write(f'{CSI}{count}P')
def screen_push_colors(which: int) -> None:
write(f'{CSI}{which}#P')
def screen_pop_colors(which: int) -> None:
write(f'{CSI}{which}#Q')
def screen_report_colors() -> None:
write(f'{CSI}#R')
def screen_repeat_character(num: int) -> None:
write(f'{CSI}{num}b')
def screen_insert_characters(count: int) -> None:
write(f'{CSI}{count}@')
def screen_scroll(count: int) -> None:
write(f'{CSI}{count}S')
def screen_erase_in_display(how: int, private: bool) -> None:
write(f'{CSI}{"?" if private else ""}{how}J')
def screen_erase_in_line(how: int, private: bool) -> None:
write(f'{CSI}{"?" if private else ""}{how}K')
def screen_erase_characters(num: int) -> None:
write(f'{CSI}{num}X')
def screen_delete_lines(num: int) -> None:
write(f'{CSI}{num}M')
def screen_cursor_up2(count: int) -> None:
write(f'{CSI}{count}A')
def screen_cursor_down(count: int) -> None:
write(f'{CSI}{count}B')
def screen_cursor_down1(count: int) -> None:
write(f'{CSI}{count}E')
def screen_report_key_encoding_flags() -> None:
write(f'{CSI}?u')
def screen_set_key_encoding_flags(flags: int, how: int) -> None:
write(f'{CSI}={flags};{how}u')
def screen_push_key_encoding_flags(flags: int) -> None:
write(f'{CSI}>{flags}u')
def screen_pop_key_encoding_flags(flags: int) -> None:
write(f'{CSI}<{flags}u')
def screen_carriage_return() -> None:
write('\r')
def screen_linefeed() -> None:
write('\n')
def screen_tab() -> None:
write('\t')
def screen_backspace() -> None:
write('\x08')
def screen_set_cursor(mode: int, secondary: int) -> None:
write(f'{CSI}{secondary} q')
def screen_insert_lines(num: int) -> None:
write(f'{CSI}{num}L')
def draw(*a: str) -> None:
write(' '.join(a))
def screen_manipulate_title_stack(op: int, which: int) -> None:
write(f'{CSI}{op};{which}t')
def report_device_attributes(mode: int, char: int) -> None:
x = CSI
if char:
x += chr(char)
if mode:
x += str(mode)
write(f'{x}c')
def report_device_status(x: int, private: bool) -> None:
write(f'{CSI}{"?" if private else ""}{x}n')
def screen_decsace(mode: int) -> None:
write(f'{CSI}{mode}*x')
def screen_set_8bit_controls(mode: int) -> None:
write(f'\x1b {"G" if mode else "F"}')
def write_osc(code: int, string: str = '') -> None:
if string:
write(f'{OSC}{code};{string}\x07')
else:
write(f'{OSC}{code}\x07')
set_color_table_color = process_cwd_notification = write_osc
clipboard_control_pending: str = ''
def set_dynamic_color(payload: str) -> None:
code, data = payload.partition(' ')[::2]
write_osc(int(code), data)
def shell_prompt_marking(payload: str) -> None:
write_osc(133, payload)
def clipboard_control(payload: str) -> None:
global clipboard_control_pending
code, data = payload.split(';', 1)
if code == '-52':
if clipboard_control_pending:
clipboard_control_pending += data.lstrip(';')
else:
clipboard_control_pending = payload
return
if clipboard_control_pending:
clipboard_control_pending += data.lstrip(';')
payload = clipboard_control_pending
clipboard_control_pending = ''
write(f'{OSC}{payload}\x07')
def multicell_command(payload: str) -> None:
c = json.loads(payload)
text = c.pop('', '')
m = ''
if (w := c.pop('width', None)) is not None and w > 0:
m += f'w={w}:'
if (s := c.pop('scale', None)) is not None and s > 1:
m += f's={s}:'
if (n := c.pop('subscale_n', None)) is not None and n > 0:
m += f'n={n}:'
if (d := c.pop('subscale_d', None)) is not None and d > 0:
m += f'd={d}:'
if (v := c.pop('vertical_align', None)) is not None and v > 0:
m += f'v={v}:'
if (h := c.pop('horizontal_align', None)) is not None and h > 0:
m += f'h={h}:'
if c:
raise Exception('Unknown keys in multicell_command: ' + ', '.join(c))
write(f'{OSC}{TEXT_SIZE_CODE};{m.rstrip(":")};{text}\a')
def screen_multi_cursor(rest: str) -> None:
write(f'{CSI}>{rest.strip()} q')
def replay(raw: str) -> None:
specials = frozenset({
'draw', 'set_title', 'set_icon', 'set_dynamic_color', 'set_color_table_color', 'select_graphic_rendition',
'process_cwd_notification', 'clipboard_control', 'shell_prompt_marking', 'multicell_command', 'screen_multi_cursor',
})
for line in raw.splitlines():
if line.strip() and not line.startswith('#'):
cmd, rest = line.partition(' ')[::2]
if cmd in specials:
globals()[cmd](rest)
else:
r = map(int, rest.split()) if rest else ()
globals()[cmd](*r)
def main(path: str) -> None:
with open(path) as f:
raw = f.read()
replay(raw)
with suppress(EOFError, KeyboardInterrupt):
input()