subiquity/subiquitycore/view.py

131 lines
3.9 KiB
Python
Raw Normal View History

# Copyright 2015 Canonical, Ltd.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
""" View policy
Contains some default key navigations
"""
import logging
from urwid import (
emit_signal,
Overlay,
Text,
)
2018-06-21 21:38:18 +00:00
from subiquitycore.ui.container import (
Columns,
Pile,
WidgetWrap,
)
from subiquitycore.ui.stretchy import StretchyOverlay
from subiquitycore.ui.utils import disabled, undisabled
log = logging.getLogger('subiquitycore.view')
class OverlayNotFoundError(Exception):
""" Exception to raise when trying to remove a non-existent overlay. """
class BaseView(WidgetWrap):
2019-09-25 23:56:24 +00:00
def local_help(self):
"""Help for what the user is currently looking at.
Returns title, documentation (or None, None).
"""
return None, None
2017-01-06 07:44:35 +00:00
def show_overlay(self, overlay_widget, **kw):
args = dict(
align='center',
width=('relative', 60),
min_width=80,
valign='middle',
height='pack'
)
PADDING = 3
2018-05-22 16:42:22 +00:00
# Don't expect callers to account for the padding if
# they pass a fixed width.
if 'width' in kw:
if isinstance(kw['width'], int):
kw['width'] += 2*PADDING
2017-01-06 07:44:35 +00:00
args.update(kw)
top = Pile([
('pack', Text("")),
Columns([
(PADDING, Text("")),
overlay_widget,
(PADDING, Text(""))
]),
('pack', Text("")),
])
self._w = Overlay(top_w=top, bottom_w=disabled(self._w), **args)
2017-01-06 07:44:35 +00:00
def show_stretchy_overlay(self, stretchy):
emit_signal(stretchy, 'opened')
stretchy.opened()
self._w = StretchyOverlay(disabled(self._w), stretchy)
def remove_overlay(self, stretchy=None,
*, not_found_ok=False) -> None:
""" Remove (frontmost) overlay from the view. """
if stretchy is not None:
one_above = None
cur = self._w
while isinstance(cur, (StretchyOverlay, Overlay)):
cur_stretchy = getattr(cur, 'stretchy', None)
if cur_stretchy is stretchy:
emit_signal(stretchy, 'closed')
stretchy.closed()
if one_above is not None:
one_above.bottom_w = cur.bottom_w
else:
self._w = undisabled(cur.bottom_w)
return
one_above = cur
cur = undisabled(cur.bottom_w)
else:
if not not_found_ok:
raise OverlayNotFoundError
else:
try:
behind_overlay = self._w.bottom_w
except AttributeError:
if not_found_ok:
return
raise OverlayNotFoundError
if isinstance(self._w, StretchyOverlay):
emit_signal(self._w.stretchy, 'closed')
self._w.stretchy.closed()
self._w = undisabled(behind_overlay)
2017-01-06 07:44:35 +00:00
def cancel(self):
pass
def keypress(self, size, key):
2017-02-12 23:18:48 +00:00
key = super().keypress(size, key)
if key == 'esc':
try:
self.remove_overlay(not_found_ok=False)
except OverlayNotFoundError:
self.cancel()
return None
2017-02-12 23:18:48 +00:00
return key