server: Initial typing notice sending support.

This commit is contained in:
Damir Jelić 2018-11-29 15:54:13 +01:00
parent c8968020a9
commit 1298a2c910
4 changed files with 105 additions and 3 deletions

22
main.py
View file

@ -491,6 +491,25 @@ def lazy_fetch_members_signal(_, _signal, buffer_ptr):
return W.WEECHAT_RC_OK return W.WEECHAT_RC_OK
def typing_notification_cb(data, signal, buffer_ptr):
"""Send out typing notifications if the user is typing.
This function is called every time the input text is changed.
It checks if we are on a buffer we own, and if we are sends out a typing
notification if the room is configured to send them out.
"""
for server in SERVERS.values():
room_buffer = server.find_room_from_ptr(buffer_ptr)
if room_buffer:
server.room_send_typing_notice(room_buffer)
return W.WEECHAT_RC_OK
if buffer_ptr == server.server_buffer:
return W.WEECHAT_RC_OK
return W.WEECHAT_RC_OK
if __name__ == "__main__": if __name__ == "__main__":
if W.register(WEECHAT_SCRIPT_NAME, WEECHAT_SCRIPT_AUTHOR, if W.register(WEECHAT_SCRIPT_NAME, WEECHAT_SCRIPT_AUTHOR,
WEECHAT_SCRIPT_VERSION, WEECHAT_SCRIPT_LICENSE, WEECHAT_SCRIPT_VERSION, WEECHAT_SCRIPT_LICENSE,
@ -513,7 +532,8 @@ if __name__ == "__main__":
init_bar_items() init_bar_items()
init_completion() init_completion()
hook = W.hook_signal("buffer_switch", "lazy_fetch_members_signal", "") W.hook_signal("buffer_switch", "lazy_fetch_members_signal", "")
W.hook_signal("input_text_changed", "typing_notification_cb", "")
if not SERVERS: if not SERVERS:
create_default_server(G.CONFIG) create_default_server(G.CONFIG)

View file

@ -49,7 +49,7 @@ from nio import (
from . import globals as G from . import globals as G
from .colors import Formatted from .colors import Formatted
from .config import RedactType from .config import RedactType
from .globals import SCRIPT_NAME, SERVERS, W from .globals import SCRIPT_NAME, SERVERS, W, TYPING_NOTICE_TIMEOUT
from .utf import utf8_decode from .utf import utf8_decode
from .utils import server_ts_to_weechat, shorten_sender, string_strikethrough from .utils import server_ts_to_weechat, shorten_sender, string_strikethrough
@ -414,6 +414,12 @@ class WeechatChannelBuffer(object):
self.remove_smart_filtered_nick(nick) self.remove_smart_filtered_nick(nick)
@property
def input(self):
# type: () -> str
"""Get the bar item input text of the buffer."""
return W.buffer_get_string(self._ptr, "input")
@property @property
def lines(self): def lines(self):
own_lines = W.hdata_pointer(self._hdata, self._ptr, "own_lines") own_lines = W.hdata_pointer(self._hdata, self._ptr, "own_lines")
@ -819,6 +825,9 @@ class RoomBuffer(object):
self.printed_before_ack_queue = list() # type: List[UUID] self.printed_before_ack_queue = list() # type: List[UUID]
self.undecrypted_events = deque(maxlen=5000) self.undecrypted_events = deque(maxlen=5000)
self.typing_notice_time = None
self._typing = False
buffer_name = "{}.{}".format(server_name, room.room_id) buffer_name = "{}.{}".format(server_name, room.room_id)
# This dict remembers the connection from a user_id to the name we # This dict remembers the connection from a user_id to the name we
@ -860,6 +869,35 @@ class RoomBuffer(object):
def warning_prefix(self): def warning_prefix(self):
return "⚠️ " return "⚠️ "
@property
def typing(self):
# type: () -> bool
"""Return our typing status."""
return self._typing
@typing.setter
def typing(self, value):
self._typing = value
if value:
self.typing_notice_time = time.time()
else:
self.typing_notice_time = None
@property
def typing_notice_expired(self):
# type: () -> bool
"""Check if the typing notice has expired.
Returns true if a new typing notice should be sent.
"""
if not self.typing_notice_time:
return True
now = time.time()
if (now - self.typing_notice_time) > (TYPING_NOTICE_TIMEOUT / 1000):
return True
return False
def find_nick(self, user_id): def find_nick(self, user_id):
# type: (str) -> str # type: (str) -> str
"""Find a suitable nick from a user_id""" """Find a suitable nick from a user_id"""

View file

@ -40,3 +40,4 @@ CONFIG = None # type: Optional[MatrixConfig]
ENCRYPTION = True # type: bool ENCRYPTION = True # type: bool
SCRIPT_NAME = "matrix" # type: str SCRIPT_NAME = "matrix" # type: str
MAX_EVENTS = 100 MAX_EVENTS = 100
TYPING_NOTICE_TIMEOUT = 4000 # 4 seconds typing notice lifetime

View file

@ -56,7 +56,7 @@ from nio import (
from . import globals as G from . import globals as G
from .buffer import OwnAction, OwnMessage, RoomBuffer from .buffer import OwnAction, OwnMessage, RoomBuffer
from .config import ConfigSection, Option, ServerBufferType from .config import ConfigSection, Option, ServerBufferType
from .globals import SCRIPT_NAME, SERVERS, W, MAX_EVENTS from .globals import SCRIPT_NAME, SERVERS, W, MAX_EVENTS, TYPING_NOTICE_TIMEOUT
from .utf import utf8_decode from .utf import utf8_decode
from .utils import create_server_buffer, key_from_value, server_buffer_prnt from .utils import create_server_buffer, key_from_value, server_buffer_prnt
@ -673,6 +673,49 @@ class MatrixServer(object):
self.backlog_queue[uuid] = room_id self.backlog_queue[uuid] = room_id
self.send_or_queue(request) self.send_or_queue(request)
def room_send_typing_notice(self, room_buffer):
"""Send a typing notice for the provided room.
Args:
room_buffer(RoomBuffer): the room for which the typing notice needs
to be sent.
"""
if not self.connected:
return
input = room_buffer.weechat_buffer.input
# Don't send a typing notice if the user is typing in a weechat command
if input.startswith("/") and not input.startswith("//"):
return
# Don't send a typing notice if we only typed a couple of letters.
elif len(input) < 4 and not room_buffer.typing:
return
# If we were typing already and our input bar now has no letters or
# only a couple of letters stop the typing notice.
elif len(input) < 4:
_, request = self.client.room_typing(
room_buffer.room.room_id,
typing_state=False)
room_buffer.typing = False
self.send(request)
return
# Don't send out a typing notice if we already sent one out and it
# didn't expire yet.
if not room_buffer.typing_notice_expired:
return
_, request = self.client.room_typing(
room_buffer.room.room_id,
typing_state=True,
timeout=TYPING_NOTICE_TIMEOUT)
room_buffer.typing = True
self.send(request)
def room_send_message( def room_send_message(
self, self,
room_buffer, # type: RoomBuffer room_buffer, # type: RoomBuffer