Skip to content

Commit f8d1ee5

Browse files
committed
ui: Periodically cycle random footer hint.
Switch to using urwid's alarm instead of time.sleep(), as that timer can be cancelled when a new footer event arrives. Update test using freezegun.
1 parent bec6cb1 commit f8d1ee5

File tree

2 files changed

+21
-10
lines changed

2 files changed

+21
-10
lines changed

tests/ui/test_ui.py

+14-6
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
from typing import Any, Callable, List, Optional
22

33
import pytest
4+
from freezegun import freeze_time # type: ignore [import-untyped]
45
from pytest_mock import MockerFixture
56
from urwid import Widget
67

@@ -109,14 +110,21 @@ def test_set_footer_text_with_duration(
109110
duration: Optional[float] = 5.3,
110111
) -> None:
111112
mocker.patch(VIEW + ".get_random_help", return_value=["some help text"])
112-
mock_sleep = mocker.patch("time.sleep")
113+
mocked_alarm = mocker.patch(CONTROLLER + ".loop.set_alarm_in")
113114

114-
view.set_footer_text([custom_text], duration=duration)
115+
with freeze_time() as frozen_datetime:
116+
view.set_footer_text([custom_text], duration=duration)
115117

116-
view.frame.footer.set_text.assert_has_calls(
117-
[mocker.call([custom_text]), mocker.call(["some help text"])]
118-
)
119-
mock_sleep.assert_called_once_with(duration)
118+
view.frame.footer.set_text.assert_called_once_with([custom_text])
119+
assert view.controller.update_screen.call_count == 1
120+
121+
frozen_datetime.tick(duration)
122+
123+
# Trigger the alarm
124+
alarm_callback = mocked_alarm.call_args[0][1]
125+
alarm_callback(None, None)
126+
127+
view.frame.footer.set_text.assert_called_with(["some help text"])
120128
assert view.controller.update_screen.call_count == 2
121129

122130
@pytest.mark.parametrize(

zulipterminal/ui.py

+7-4
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,6 @@
44

55
import random
66
import re
7-
import time
87
from typing import Any, Dict, List, Optional
98

109
import urwid
@@ -120,7 +119,7 @@ def set_footer_text(
120119
self,
121120
text_list: Optional[List[Any]] = None,
122121
style: str = "footer",
123-
duration: Optional[float] = None,
122+
duration: Optional[float] = 30,
124123
) -> None:
125124
# Avoid updating repeatedly (then pausing and showing default text)
126125
# This is simple, though doesn't avoid starting one thread for each call
@@ -134,10 +133,14 @@ def set_footer_text(
134133
self.frame.footer.set_text(text)
135134
self.frame.footer.set_attr_map({None: style})
136135
self.controller.update_screen()
136+
137+
if hasattr(self, "_footer_alarm_handle"):
138+
self.controller.loop.remove_alarm(self._footer_alarm_handle)
137139
if duration is not None:
138140
assert duration > 0
139-
time.sleep(duration)
140-
self.set_footer_text()
141+
self._footer_alarm_handle: Any = self.controller.loop.set_alarm_in(
142+
duration, lambda loop, user_data: self.set_footer_text()
143+
)
141144

142145
@asynch
143146
def set_typeahead_footer(

0 commit comments

Comments
 (0)