2024-05-05 21:32:23 +02:00
|
|
|
# Copyright 2000-2004 Michael Hudson-Doyle <micahel@gmail.com>
|
|
|
|
|
#
|
|
|
|
|
# All Rights Reserved
|
|
|
|
|
#
|
|
|
|
|
#
|
|
|
|
|
# Permission to use, copy, modify, and distribute this software and
|
|
|
|
|
# its documentation for any purpose is hereby granted without fee,
|
|
|
|
|
# provided that the above copyright notice appear in all copies and
|
|
|
|
|
# that both that copyright notice and this permission notice appear in
|
|
|
|
|
# supporting documentation.
|
|
|
|
|
#
|
|
|
|
|
# THE AUTHOR MICHAEL HUDSON DISCLAIMS ALL WARRANTIES WITH REGARD TO
|
|
|
|
|
# THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
|
|
|
# AND FITNESS, IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL,
|
|
|
|
|
# INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
|
|
|
|
|
# RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF
|
|
|
|
|
# CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
|
|
|
|
# CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
|
|
|
|
|
|
import termios
|
|
|
|
|
|
|
|
|
|
|
2025-09-17 11:59:49 +01:00
|
|
|
TYPE_CHECKING = False
|
|
|
|
|
|
|
|
|
|
if TYPE_CHECKING:
|
|
|
|
|
from typing import cast
|
|
|
|
|
else:
|
|
|
|
|
cast = lambda typ, val: val
|
|
|
|
|
|
|
|
|
|
|
2024-05-05 21:32:23 +02:00
|
|
|
class TermState:
|
2025-09-17 11:59:49 +01:00
|
|
|
def __init__(self, attrs: list[int | list[bytes]]) -> None:
|
|
|
|
|
self.iflag = cast(int, attrs[0])
|
|
|
|
|
self.oflag = cast(int, attrs[1])
|
|
|
|
|
self.cflag = cast(int, attrs[2])
|
|
|
|
|
self.lflag = cast(int, attrs[3])
|
|
|
|
|
self.ispeed = cast(int, attrs[4])
|
|
|
|
|
self.ospeed = cast(int, attrs[5])
|
|
|
|
|
self.cc = cast(list[bytes], attrs[6])
|
2024-05-05 21:32:23 +02:00
|
|
|
|
2025-09-17 11:59:49 +01:00
|
|
|
def as_list(self) -> list[int | list[bytes]]:
|
2024-05-05 21:32:23 +02:00
|
|
|
return [
|
|
|
|
|
self.iflag,
|
|
|
|
|
self.oflag,
|
|
|
|
|
self.cflag,
|
|
|
|
|
self.lflag,
|
|
|
|
|
self.ispeed,
|
|
|
|
|
self.ospeed,
|
2025-01-23 17:30:08 +00:00
|
|
|
# Always return a copy of the control characters list to ensure
|
|
|
|
|
# there are not any additional references to self.cc
|
|
|
|
|
self.cc[:],
|
2024-05-05 21:32:23 +02:00
|
|
|
]
|
|
|
|
|
|
2025-09-17 11:59:49 +01:00
|
|
|
def copy(self) -> "TermState":
|
2024-05-05 21:32:23 +02:00
|
|
|
return self.__class__(self.as_list())
|
|
|
|
|
|
|
|
|
|
|
2025-09-17 11:59:49 +01:00
|
|
|
def tcgetattr(fd: int) -> TermState:
|
2024-05-05 21:32:23 +02:00
|
|
|
return TermState(termios.tcgetattr(fd))
|
|
|
|
|
|
|
|
|
|
|
2025-09-17 11:59:49 +01:00
|
|
|
def tcsetattr(fd: int, when: int, attrs: TermState) -> None:
|
2024-05-05 21:32:23 +02:00
|
|
|
termios.tcsetattr(fd, when, attrs.as_list())
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Term(TermState):
|
|
|
|
|
TS__init__ = TermState.__init__
|
|
|
|
|
|
2025-09-17 11:59:49 +01:00
|
|
|
def __init__(self, fd: int = 0) -> None:
|
2024-05-05 21:32:23 +02:00
|
|
|
self.TS__init__(termios.tcgetattr(fd))
|
|
|
|
|
self.fd = fd
|
2025-09-17 11:59:49 +01:00
|
|
|
self.stack: list[list[int | list[bytes]]] = []
|
2024-05-05 21:32:23 +02:00
|
|
|
|
2025-09-17 11:59:49 +01:00
|
|
|
def save(self) -> None:
|
2024-05-05 21:32:23 +02:00
|
|
|
self.stack.append(self.as_list())
|
|
|
|
|
|
2025-09-17 11:59:49 +01:00
|
|
|
def set(self, when: int = termios.TCSANOW) -> None:
|
2024-05-05 21:32:23 +02:00
|
|
|
termios.tcsetattr(self.fd, when, self.as_list())
|
|
|
|
|
|
2025-09-17 11:59:49 +01:00
|
|
|
def restore(self) -> None:
|
2024-05-05 21:32:23 +02:00
|
|
|
self.TS__init__(self.stack.pop())
|
|
|
|
|
self.set()
|