2021-03-11 19:16:15 +01:00
|
|
|
"""Main browser logic."""
|
|
|
|
|
2021-02-12 19:01:42 +01:00
|
|
|
import curses
|
|
|
|
import curses.ascii
|
|
|
|
import curses.textpad
|
|
|
|
import os
|
2021-03-13 20:37:13 +01:00
|
|
|
import webbrowser
|
2021-02-18 19:01:28 +01:00
|
|
|
from math import inf
|
2021-02-12 19:01:42 +01:00
|
|
|
|
|
|
|
from bebop.colors import ColorPair, init_colors
|
2021-03-13 20:37:13 +01:00
|
|
|
from bebop.command_line import CommandLine
|
2021-02-18 01:40:05 +01:00
|
|
|
from bebop.history import History
|
2021-03-13 16:31:11 +01:00
|
|
|
from bebop.links import Links
|
2021-02-12 19:01:42 +01:00
|
|
|
from bebop.mouse import ButtonState
|
2021-02-16 19:10:11 +01:00
|
|
|
from bebop.navigation import join_url, parse_url, sanitize_url, set_parameter
|
2021-03-13 20:37:13 +01:00
|
|
|
from bebop.page import Page, PagePad
|
2021-02-12 19:01:42 +01:00
|
|
|
from bebop.protocol import Request, Response
|
|
|
|
|
|
|
|
|
2021-02-16 21:22:49 +01:00
|
|
|
class Browser:
|
|
|
|
"""Manage the events, inputs and rendering."""
|
2021-02-12 19:01:42 +01:00
|
|
|
|
|
|
|
def __init__(self, cert_stash):
|
2021-03-13 16:31:11 +01:00
|
|
|
self.stash = cert_stash or {}
|
2021-02-12 19:01:42 +01:00
|
|
|
self.screen = None
|
|
|
|
self.dim = (0, 0)
|
2021-03-13 20:37:13 +01:00
|
|
|
self.page_pad = None
|
2021-02-12 23:29:51 +01:00
|
|
|
self.status_line = None
|
|
|
|
self.command_line = None
|
2021-03-13 20:37:13 +01:00
|
|
|
self.running = True
|
2021-03-09 00:44:54 +01:00
|
|
|
self.status_data = ("", 0, 0)
|
2021-02-12 23:29:51 +01:00
|
|
|
self.current_url = ""
|
2021-02-18 01:40:05 +01:00
|
|
|
self.history = History()
|
2021-03-14 00:05:22 +01:00
|
|
|
self.cache = {}
|
2021-02-12 19:01:42 +01:00
|
|
|
|
|
|
|
@property
|
|
|
|
def h(self):
|
|
|
|
return self.dim[0]
|
|
|
|
|
|
|
|
@property
|
|
|
|
def w(self):
|
|
|
|
return self.dim[1]
|
|
|
|
|
|
|
|
def run(self, *args, **kwargs):
|
|
|
|
"""Use curses' wrapper around _run."""
|
|
|
|
os.environ.setdefault("ESCDELAY", "25")
|
|
|
|
curses.wrapper(self._run, *args, **kwargs)
|
|
|
|
|
|
|
|
def _run(self, stdscr, start_url=None):
|
|
|
|
"""Start displaying content and handling events."""
|
|
|
|
self.screen = stdscr
|
|
|
|
self.screen.clear()
|
|
|
|
self.screen.refresh()
|
2021-02-12 23:29:51 +01:00
|
|
|
|
2021-02-12 19:01:42 +01:00
|
|
|
curses.mousemask(curses.ALL_MOUSE_EVENTS)
|
2021-02-12 23:29:51 +01:00
|
|
|
curses.curs_set(0)
|
|
|
|
init_colors()
|
2021-02-12 19:01:42 +01:00
|
|
|
|
|
|
|
self.dim = self.screen.getmaxyx()
|
2021-03-13 20:37:13 +01:00
|
|
|
self.page_pad = PagePad(self.h - 2)
|
2021-02-12 23:29:51 +01:00
|
|
|
self.status_line = self.screen.subwin(
|
2021-02-12 19:01:42 +01:00
|
|
|
*self.line_dim,
|
2021-02-12 23:29:51 +01:00
|
|
|
*self.status_line_pos,
|
2021-02-12 19:01:42 +01:00
|
|
|
)
|
2021-02-12 23:29:51 +01:00
|
|
|
command_line_window = self.screen.subwin(
|
2021-02-12 19:01:42 +01:00
|
|
|
*self.line_dim,
|
2021-02-12 23:29:51 +01:00
|
|
|
*self.command_line_pos,
|
2021-02-12 19:01:42 +01:00
|
|
|
)
|
2021-02-12 23:29:51 +01:00
|
|
|
self.command_line = CommandLine(command_line_window)
|
2021-02-12 19:01:42 +01:00
|
|
|
|
2021-03-08 23:40:03 +01:00
|
|
|
if start_url:
|
|
|
|
self.open_url(start_url, assume_absolute=True)
|
2021-02-12 19:01:42 +01:00
|
|
|
|
2021-03-08 23:40:03 +01:00
|
|
|
while self.running:
|
2021-03-09 00:44:42 +01:00
|
|
|
try:
|
|
|
|
self.handle_inputs()
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
self.set_status("Cancelled.")
|
|
|
|
|
|
|
|
def handle_inputs(self):
|
|
|
|
char = self.screen.getch()
|
|
|
|
if char == ord(":"):
|
|
|
|
self.quick_command("")
|
|
|
|
elif char == ord("r"):
|
|
|
|
self.reload_page()
|
|
|
|
elif char == ord("h"):
|
|
|
|
self.scroll_page_horizontally(-3)
|
2021-03-13 18:58:02 +01:00
|
|
|
elif char == ord("H"):
|
|
|
|
pass # TODO h-scroll whole page left
|
2021-03-09 00:44:42 +01:00
|
|
|
elif char == ord("j"):
|
|
|
|
self.scroll_page_vertically(3)
|
2021-03-13 18:58:02 +01:00
|
|
|
elif char == ord("J"):
|
|
|
|
self.scroll_whole_page_down()
|
2021-03-09 00:44:42 +01:00
|
|
|
elif char == ord("k"):
|
|
|
|
self.scroll_page_vertically(-3)
|
2021-03-13 18:58:02 +01:00
|
|
|
elif char == ord("K"):
|
|
|
|
self.scroll_whole_page_up()
|
2021-03-09 00:44:42 +01:00
|
|
|
elif char == ord("l"):
|
|
|
|
self.scroll_page_horizontally(3)
|
2021-03-13 18:58:02 +01:00
|
|
|
elif char == ord("L"):
|
|
|
|
pass # TODO h-scroll whole page right
|
|
|
|
elif char == ord("^"):
|
|
|
|
pass # TODO reset horizontal scrolling
|
2021-03-09 00:44:42 +01:00
|
|
|
elif char == ord("g"):
|
2021-02-12 19:01:42 +01:00
|
|
|
char = self.screen.getch()
|
2021-03-09 00:44:42 +01:00
|
|
|
if char == ord("g"):
|
|
|
|
self.scroll_page_vertically(-inf)
|
|
|
|
elif char == ord("G"):
|
|
|
|
self.scroll_page_vertically(inf)
|
2021-03-13 18:58:02 +01:00
|
|
|
elif char == ord("o"):
|
|
|
|
self.quick_command("open")
|
|
|
|
elif char == ord("p"):
|
|
|
|
self.go_back()
|
2021-03-09 00:44:42 +01:00
|
|
|
elif curses.ascii.isdigit(char):
|
|
|
|
self.handle_digit_input(char)
|
|
|
|
elif char == curses.KEY_MOUSE:
|
|
|
|
self.handle_mouse(*curses.getmouse())
|
|
|
|
elif char == curses.KEY_RESIZE:
|
|
|
|
self.handle_resize()
|
|
|
|
elif char == curses.ascii.ESC: # Can be ESC or ALT char.
|
|
|
|
self.screen.nodelay(True)
|
2021-03-13 18:58:02 +01:00
|
|
|
char = self.screen.getch()
|
|
|
|
if char == -1:
|
2021-03-09 00:44:42 +01:00
|
|
|
self.set_status(self.current_url)
|
2021-03-13 18:58:02 +01:00
|
|
|
else: # ALT keybinds.
|
|
|
|
if char == ord("h"):
|
|
|
|
self.scroll_page_horizontally(-1)
|
|
|
|
elif char == ord("j"):
|
|
|
|
self.scroll_page_vertically(1)
|
|
|
|
elif char == ord("k"):
|
|
|
|
self.scroll_page_vertically(-1)
|
|
|
|
elif char == ord("l"):
|
|
|
|
self.scroll_page_horizontally(1)
|
2021-03-09 00:44:42 +01:00
|
|
|
self.screen.nodelay(False)
|
2021-03-14 00:04:55 +01:00
|
|
|
# else:
|
|
|
|
# unctrled = curses.unctrl(char)
|
|
|
|
# if unctrled == b"^T":
|
|
|
|
# self.set_status("test!")
|
2021-03-13 18:58:02 +01:00
|
|
|
|
2021-02-12 19:01:42 +01:00
|
|
|
@property
|
2021-02-12 23:29:51 +01:00
|
|
|
def page_pad_size(self):
|
2021-02-12 19:01:42 +01:00
|
|
|
return self.h - 3, self.w - 1
|
|
|
|
|
|
|
|
@property
|
2021-02-12 23:29:51 +01:00
|
|
|
def status_line_pos(self):
|
2021-02-12 19:01:42 +01:00
|
|
|
return self.h - 2, 0
|
|
|
|
|
|
|
|
@property
|
2021-02-12 23:29:51 +01:00
|
|
|
def command_line_pos(self):
|
2021-02-12 19:01:42 +01:00
|
|
|
return self.h - 1, 0
|
|
|
|
|
|
|
|
@property
|
|
|
|
def line_dim(self):
|
|
|
|
return 1, self.w
|
|
|
|
|
|
|
|
def refresh_windows(self):
|
2021-02-16 21:22:49 +01:00
|
|
|
"""Refresh all windows and clear command line."""
|
2021-02-12 23:29:51 +01:00
|
|
|
self.refresh_page()
|
|
|
|
self.refresh_status_line()
|
|
|
|
self.command_line.clear()
|
|
|
|
|
|
|
|
def refresh_page(self):
|
2021-02-16 21:22:49 +01:00
|
|
|
"""Refresh the current page pad; it does not reload the page."""
|
2021-03-13 20:37:13 +01:00
|
|
|
self.page_pad.refresh_content(*self.page_pad_size)
|
2021-02-12 19:01:42 +01:00
|
|
|
|
2021-02-12 23:29:51 +01:00
|
|
|
def refresh_status_line(self):
|
2021-02-12 19:01:42 +01:00
|
|
|
"""Refresh status line contents."""
|
2021-03-09 00:44:54 +01:00
|
|
|
text, pair, attributes = self.status_data
|
2021-02-12 19:01:42 +01:00
|
|
|
text = text[:self.w - 1]
|
2021-03-09 00:44:54 +01:00
|
|
|
color = curses.color_pair(pair)
|
|
|
|
self.status_line.addstr(0, 0, text, color | attributes)
|
2021-02-12 23:29:51 +01:00
|
|
|
self.status_line.clrtoeol()
|
|
|
|
self.status_line.refresh()
|
2021-02-12 19:01:42 +01:00
|
|
|
|
|
|
|
def set_status(self, text):
|
|
|
|
"""Set a regular message in the status bar."""
|
2021-03-09 00:44:54 +01:00
|
|
|
self.status_data = text, ColorPair.NORMAL, curses.A_ITALIC
|
2021-02-12 23:29:51 +01:00
|
|
|
self.refresh_status_line()
|
2021-02-12 19:01:42 +01:00
|
|
|
|
|
|
|
def set_status_error(self, text):
|
|
|
|
"""Set an error message in the status bar."""
|
2021-03-09 00:44:54 +01:00
|
|
|
self.status_data = text, ColorPair.ERROR, 0
|
2021-02-12 23:29:51 +01:00
|
|
|
self.refresh_status_line()
|
2021-02-12 19:01:42 +01:00
|
|
|
|
2021-03-13 16:31:11 +01:00
|
|
|
def quick_command(self, command):
|
|
|
|
"""Shortcut method to take user input with a prefixed command string."""
|
|
|
|
prefix = f"{command} " if command else ""
|
|
|
|
user_input = self.command_line.focus(":", prefix=prefix)
|
|
|
|
if not user_input:
|
|
|
|
return
|
|
|
|
self.process_command(user_input)
|
|
|
|
|
|
|
|
def process_command(self, command_text: str):
|
|
|
|
"""Handle a client command."""
|
|
|
|
words = command_text.split()
|
|
|
|
num_words = len(words)
|
|
|
|
if num_words == 0:
|
|
|
|
return
|
|
|
|
command = words[0]
|
|
|
|
if num_words == 1:
|
|
|
|
if command in ("q", "quit"):
|
|
|
|
self.running = False
|
|
|
|
return
|
|
|
|
if command in ("o", "open"):
|
|
|
|
self.open_url(words[1], assume_absolute=True)
|
|
|
|
|
2021-02-26 17:41:16 +01:00
|
|
|
def open_url(self, url, base_url=None, redirects=0, assume_absolute=False):
|
2021-02-12 19:01:42 +01:00
|
|
|
"""Try to open an URL.
|
|
|
|
|
2021-02-16 21:22:49 +01:00
|
|
|
This function assumes that the URL can be from an user and thus tries a
|
|
|
|
few things to make it work.
|
|
|
|
|
|
|
|
If there is no current URL (e.g. we just started) or `assume_absolute`
|
|
|
|
is True, assume it is an absolute URL. In other cases, parse it normally
|
|
|
|
and later check if it has to be used relatively to the current URL.
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
- url: an URL string, may not be completely compliant.
|
2021-02-26 17:41:16 +01:00
|
|
|
- base_url: an URL string to use as base in case `url` is relative.
|
2021-02-16 21:22:49 +01:00
|
|
|
- redirections: number of redirections we did yet for the same request.
|
|
|
|
- assume_absolute: assume we intended to use an absolute URL if True.
|
2021-02-12 19:01:42 +01:00
|
|
|
"""
|
2021-02-26 17:41:16 +01:00
|
|
|
if redirects > 5:
|
2021-02-15 19:57:49 +01:00
|
|
|
self.set_status_error(f"Too many redirections ({url}).")
|
2021-02-13 23:34:45 +01:00
|
|
|
return
|
2021-02-16 21:22:49 +01:00
|
|
|
if assume_absolute or not self.current_url:
|
2021-02-12 19:01:42 +01:00
|
|
|
parts = parse_url(url, absolute=True)
|
2021-02-17 01:31:24 +01:00
|
|
|
join = False
|
2021-02-16 21:22:49 +01:00
|
|
|
else:
|
|
|
|
parts = parse_url(url)
|
2021-02-17 01:31:24 +01:00
|
|
|
join = True
|
2021-02-12 19:01:42 +01:00
|
|
|
if parts.scheme == "gemini":
|
2021-02-13 23:34:45 +01:00
|
|
|
# If there is no netloc, this is a relative URL.
|
2021-02-26 17:41:16 +01:00
|
|
|
if join or base_url:
|
|
|
|
url = join_url(base_url or self.current_url, url)
|
|
|
|
self.open_gemini_url(sanitize_url(url), redirects)
|
2021-02-26 15:36:56 +01:00
|
|
|
elif parts.scheme.startswith("http"):
|
|
|
|
self.open_web_url(url)
|
2021-03-08 23:40:03 +01:00
|
|
|
elif parts.scheme == "file":
|
|
|
|
self.open_file(parts.path)
|
2021-02-12 19:01:42 +01:00
|
|
|
else:
|
2021-02-15 19:57:49 +01:00
|
|
|
self.set_status_error(f"Protocol {parts.scheme} not supported.")
|
2021-02-12 19:01:42 +01:00
|
|
|
|
2021-02-26 17:41:16 +01:00
|
|
|
def open_gemini_url(self, url, redirects=0, history=True):
|
2021-03-13 16:31:11 +01:00
|
|
|
"""Open a Gemini URL and set the formatted response as content.
|
|
|
|
|
|
|
|
After initiating the connection, TODO
|
|
|
|
"""
|
2021-02-12 19:01:42 +01:00
|
|
|
self.set_status(f"Loading {url}")
|
2021-03-14 00:05:22 +01:00
|
|
|
|
|
|
|
if url in self.cache:
|
|
|
|
self.load_page(self.cache[url])
|
|
|
|
if self.current_url and history:
|
|
|
|
self.history.push(self.current_url)
|
|
|
|
self.current_url = url
|
|
|
|
self.set_status(url)
|
|
|
|
return
|
|
|
|
|
2021-02-12 19:01:42 +01:00
|
|
|
req = Request(url, self.stash)
|
|
|
|
connected = req.connect()
|
|
|
|
if not connected:
|
|
|
|
if req.state == Request.STATE_ERROR_CERT:
|
2021-02-15 19:57:49 +01:00
|
|
|
error = f"Certificate was missing or corrupt ({url})."
|
2021-02-12 19:01:42 +01:00
|
|
|
elif req.state == Request.STATE_UNTRUSTED_CERT:
|
2021-03-13 18:58:19 +01:00
|
|
|
error = f"Certificate has been changed ({url})."
|
2021-02-12 19:01:42 +01:00
|
|
|
# TODO propose the user ways to handle this.
|
2021-02-15 19:57:49 +01:00
|
|
|
elif req.state == Request.STATE_CONNECTION_FAILED:
|
2021-03-13 18:58:19 +01:00
|
|
|
error_details = f": {req.error}" if req.error else "."
|
|
|
|
error = f"Connection failed ({url})" + error_details
|
2021-02-12 19:01:42 +01:00
|
|
|
else:
|
2021-03-13 18:58:19 +01:00
|
|
|
error = f"Connection failed ({url})."
|
|
|
|
self.set_status_error(error)
|
2021-02-12 19:01:42 +01:00
|
|
|
return
|
|
|
|
|
|
|
|
if req.state == Request.STATE_INVALID_CERT:
|
|
|
|
# TODO propose abort / temp trust
|
|
|
|
pass
|
|
|
|
elif req.state == Request.STATE_UNKNOWN_CERT:
|
|
|
|
# TODO propose abort / temp trust / perm trust
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
pass # TODO
|
|
|
|
|
2021-03-05 19:27:42 +01:00
|
|
|
data = req.proceed()
|
|
|
|
if not data:
|
|
|
|
self.set_status_error(f"Server did not respond in time ({url}).")
|
|
|
|
return
|
|
|
|
response = Response.parse(data)
|
2021-02-12 19:01:42 +01:00
|
|
|
if not response:
|
2021-02-15 19:57:49 +01:00
|
|
|
self.set_status_error(f"Server response parsing failed ({url}).")
|
2021-02-12 19:01:42 +01:00
|
|
|
return
|
|
|
|
|
2021-02-12 23:29:51 +01:00
|
|
|
if response.code == 20:
|
2021-03-14 00:04:55 +01:00
|
|
|
handle_code = self.handle_response_content(response)
|
|
|
|
if handle_code == 0:
|
|
|
|
if self.current_url and history:
|
|
|
|
self.history.push(self.current_url)
|
|
|
|
self.current_url = url
|
2021-03-14 00:05:22 +01:00
|
|
|
self.cache[url] = self.page_pad.current_page
|
2021-03-14 00:04:55 +01:00
|
|
|
self.set_status(url)
|
|
|
|
elif handle_code == 1:
|
|
|
|
self.set_status(f"Downloaded {url}.")
|
2021-02-12 23:29:51 +01:00
|
|
|
elif response.generic_code == 30 and response.meta:
|
2021-02-26 17:41:16 +01:00
|
|
|
self.open_url(response.meta, base_url=url, redirects=redirects + 1)
|
2021-02-13 23:34:45 +01:00
|
|
|
elif response.generic_code in (40, 50):
|
2021-02-18 01:41:14 +01:00
|
|
|
error = f"Server error: {response.meta or Response.code.name}"
|
2021-02-15 19:57:49 +01:00
|
|
|
self.set_status_error(error)
|
2021-02-16 19:10:11 +01:00
|
|
|
elif response.generic_code == 10:
|
|
|
|
self.handle_input_request(url, response)
|
2021-03-09 00:44:42 +01:00
|
|
|
else:
|
|
|
|
error = f"Unhandled response code {response.code}"
|
|
|
|
self.set_status_error(error)
|
2021-02-12 23:29:51 +01:00
|
|
|
|
2021-03-14 00:04:55 +01:00
|
|
|
def handle_response_content(self, response: Response) -> int:
|
|
|
|
"""Handle a response's content from a Gemini server.
|
|
|
|
|
|
|
|
According to the MIME type received or inferred, render or download the
|
|
|
|
response's content.
|
|
|
|
|
|
|
|
Currently only text/gemini content is rendered.
|
|
|
|
|
|
|
|
Arguments:
|
|
|
|
- response: a successful Response.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
An error code: 0 means a page has been loaded, so any book-keeping such
|
|
|
|
as history management can be applied; 1 means a content has been
|
|
|
|
successfully retrieved but has not been displayed (e.g. non-text
|
|
|
|
content) nor saved as a page; 2 means that the content could not be
|
|
|
|
handled, either due to bogus MIME type or MIME parameters.
|
|
|
|
"""
|
|
|
|
mime_type = response.get_mime_type()
|
|
|
|
if mime_type.main_type == "text":
|
|
|
|
if mime_type.sub_type == "gemini":
|
|
|
|
encoding = mime_type.charset
|
|
|
|
try:
|
|
|
|
text = response.content.decode(encoding, errors="replace")
|
|
|
|
except LookupError:
|
|
|
|
self.set_status_error("Unknown encoding {encoding}.")
|
|
|
|
return 2
|
|
|
|
self.load_page(Page.from_gemtext(text))
|
|
|
|
return 0
|
|
|
|
else:
|
|
|
|
pass # TODO
|
|
|
|
else:
|
|
|
|
pass # TODO
|
|
|
|
return 1
|
|
|
|
|
2021-03-13 20:37:13 +01:00
|
|
|
def load_page(self, page: Page):
|
2021-02-12 23:29:51 +01:00
|
|
|
"""Load Gemtext data as the current page."""
|
2021-03-13 20:37:13 +01:00
|
|
|
old_pad_height = self.page_pad.dim[0]
|
|
|
|
self.page_pad.show_page(page)
|
|
|
|
if self.page_pad.dim[0] < old_pad_height:
|
2021-02-12 23:29:51 +01:00
|
|
|
self.screen.clear()
|
|
|
|
self.screen.refresh()
|
|
|
|
self.refresh_windows()
|
|
|
|
else:
|
|
|
|
self.refresh_page()
|
2021-02-12 19:01:42 +01:00
|
|
|
|
|
|
|
def handle_digit_input(self, init_char: int):
|
2021-03-13 16:31:11 +01:00
|
|
|
"""Focus command-line to select the link ID to follow."""
|
2021-03-13 20:37:13 +01:00
|
|
|
if not self.page_pad or self.page_pad.current_page.links is None:
|
2021-02-18 01:41:14 +01:00
|
|
|
return
|
2021-03-13 20:37:13 +01:00
|
|
|
links = self.page_pad.current_page.links
|
2021-03-13 16:31:11 +01:00
|
|
|
err, val = self.command_line.focus_for_link_navigation(init_char, links)
|
|
|
|
if err == 0:
|
|
|
|
self.open_link(links, val) # type: ignore
|
|
|
|
elif err == 2:
|
|
|
|
self.set_status_error(val)
|
|
|
|
|
|
|
|
def open_link(self, links: Links, link_id: int):
|
2021-02-12 19:01:42 +01:00
|
|
|
"""Open the link with this link ID."""
|
2021-02-12 23:29:51 +01:00
|
|
|
if not link_id in links:
|
2021-02-15 19:57:49 +01:00
|
|
|
self.set_status_error(f"Unknown link ID {link_id}.")
|
2021-02-12 19:01:42 +01:00
|
|
|
return
|
2021-02-12 23:29:51 +01:00
|
|
|
self.open_url(links[link_id])
|
2021-02-12 19:01:42 +01:00
|
|
|
|
2021-02-16 19:10:11 +01:00
|
|
|
def handle_input_request(self, from_url: str, response: Response):
|
2021-03-13 16:31:11 +01:00
|
|
|
"""Focus command-line to pass input to the server."""
|
2021-02-16 19:10:11 +01:00
|
|
|
if response.meta:
|
|
|
|
self.set_status(f"Input needed: {response.meta}")
|
|
|
|
else:
|
|
|
|
self.set_status("Input needed:")
|
2021-03-13 16:31:11 +01:00
|
|
|
user_input = self.command_line.focus("?")
|
2021-02-16 19:10:11 +01:00
|
|
|
if user_input:
|
|
|
|
url = set_parameter(from_url, user_input)
|
|
|
|
self.open_gemini_url(url)
|
|
|
|
|
2021-02-12 19:01:42 +01:00
|
|
|
def handle_mouse(self, mouse_id: int, x: int, y: int, z: int, bstate: int):
|
|
|
|
"""Handle mouse events.
|
|
|
|
|
|
|
|
Right now, only vertical scrolling is handled.
|
|
|
|
"""
|
|
|
|
if bstate & ButtonState.SCROLL_UP:
|
2021-02-15 18:51:45 +01:00
|
|
|
self.scroll_page_vertically(-3)
|
2021-02-12 19:01:42 +01:00
|
|
|
elif bstate & ButtonState.SCROLL_DOWN:
|
2021-02-15 18:51:45 +01:00
|
|
|
self.scroll_page_vertically(3)
|
2021-02-12 19:01:42 +01:00
|
|
|
|
|
|
|
def handle_resize(self):
|
|
|
|
"""Try to not make everything collapse on resizes."""
|
|
|
|
# Refresh the whole screen before changing windows to avoid random
|
|
|
|
# blank screens.
|
|
|
|
self.screen.refresh()
|
|
|
|
old_dim = self.dim
|
|
|
|
self.dim = self.screen.getmaxyx()
|
|
|
|
# Avoid work if the resizing does not impact us.
|
|
|
|
if self.dim == old_dim:
|
|
|
|
return
|
|
|
|
# Resize windows to fit the new dimensions. Content pad will be updated
|
|
|
|
# on its own at the end of the function.
|
2021-02-12 23:29:51 +01:00
|
|
|
self.status_line.resize(*self.line_dim)
|
|
|
|
self.command_line.window.resize(*self.line_dim)
|
2021-02-12 19:01:42 +01:00
|
|
|
# Move the windows to their new position if that's still possible.
|
2021-02-12 23:29:51 +01:00
|
|
|
if self.status_line_pos[0] >= 0:
|
|
|
|
self.status_line.mvwin(*self.status_line_pos)
|
|
|
|
if self.command_line_pos[0] >= 0:
|
|
|
|
self.command_line.window.mvwin(*self.command_line_pos)
|
2021-02-12 19:01:42 +01:00
|
|
|
# If the content pad does not fit its whole place, we have to clean the
|
|
|
|
# gap between it and the status line. Refresh all screen.
|
2021-03-13 20:37:13 +01:00
|
|
|
if self.page_pad.dim[0] < self.h - 2:
|
2021-02-12 19:01:42 +01:00
|
|
|
self.screen.clear()
|
|
|
|
self.screen.refresh()
|
|
|
|
self.refresh_windows()
|
2021-02-13 23:58:49 +01:00
|
|
|
|
2021-02-18 19:01:28 +01:00
|
|
|
def scroll_page_vertically(self, by_lines):
|
2021-03-13 16:31:11 +01:00
|
|
|
"""Scroll page vertically.
|
|
|
|
|
|
|
|
If `by_lines` is an integer (positive or negative), scroll the page by
|
|
|
|
this amount of lines. If `by_lines` is one of the floats inf and -inf,
|
|
|
|
go to the end of file and beginning of file, respectively.
|
|
|
|
"""
|
2021-02-18 19:01:28 +01:00
|
|
|
window_height = self.h - 2
|
|
|
|
require_refresh = False
|
|
|
|
if by_lines == inf:
|
2021-03-13 20:37:13 +01:00
|
|
|
require_refresh = self.page_pad.go_to_end(window_height)
|
2021-02-18 19:01:28 +01:00
|
|
|
elif by_lines == -inf:
|
2021-03-13 20:37:13 +01:00
|
|
|
require_refresh = self.page_pad.go_to_beginning()
|
2021-02-18 19:01:28 +01:00
|
|
|
else:
|
2021-03-13 20:37:13 +01:00
|
|
|
require_refresh = self.page_pad.scroll_v(by_lines, window_height)
|
2021-02-18 19:01:28 +01:00
|
|
|
if require_refresh:
|
2021-02-15 18:51:45 +01:00
|
|
|
self.refresh_page()
|
|
|
|
|
2021-03-13 18:58:02 +01:00
|
|
|
def scroll_whole_page_down(self):
|
|
|
|
"""Scroll down by a whole page."""
|
|
|
|
self.scroll_page_vertically(self.page_pad_size[0])
|
|
|
|
|
|
|
|
def scroll_whole_page_up(self):
|
|
|
|
"""Scroll up by a whole page."""
|
|
|
|
self.scroll_page_vertically(-self.page_pad_size[0])
|
|
|
|
|
2021-02-18 19:01:28 +01:00
|
|
|
def scroll_page_horizontally(self, by_columns):
|
2021-03-13 16:31:11 +01:00
|
|
|
"""Scroll page horizontally."""
|
2021-03-13 20:37:13 +01:00
|
|
|
if self.page_pad.scroll_h(by_columns, self.w):
|
2021-02-15 18:51:45 +01:00
|
|
|
self.refresh_page()
|
|
|
|
|
2021-02-18 01:41:02 +01:00
|
|
|
def reload_page(self):
|
2021-03-13 16:31:11 +01:00
|
|
|
"""Reload the page, if one has been previously loaded."""
|
2021-02-18 01:41:02 +01:00
|
|
|
if self.current_url:
|
|
|
|
self.open_gemini_url(self.current_url, history=False)
|
|
|
|
|
2021-02-13 23:58:49 +01:00
|
|
|
def go_back(self):
|
|
|
|
"""Go back in history if possible."""
|
2021-02-18 01:40:05 +01:00
|
|
|
if self.history.has_links():
|
2021-02-15 18:53:21 +01:00
|
|
|
self.open_gemini_url(self.history.pop(), history=False)
|
2021-02-26 15:36:56 +01:00
|
|
|
|
|
|
|
def open_web_url(self, url):
|
|
|
|
"""Open a Web URL. Currently relies in Python's webbrowser module."""
|
|
|
|
self.set_status(f"Opening {url}")
|
2021-03-13 20:37:13 +01:00
|
|
|
webbrowser.open_new_tab(url)
|
2021-03-08 23:40:03 +01:00
|
|
|
|
2021-03-13 20:37:13 +01:00
|
|
|
def open_file(self, filepath, encoding="utf-8"):
|
2021-03-08 23:40:03 +01:00
|
|
|
"""Open a file and render it.
|
|
|
|
|
|
|
|
This should be used only on Gemtext files or at least text files.
|
2021-03-13 20:37:13 +01:00
|
|
|
Anything else will produce garbage and may crash the program. In the
|
|
|
|
future this should be able to use a different parser according to a MIME
|
|
|
|
type or something.
|
2021-03-08 23:40:03 +01:00
|
|
|
"""
|
|
|
|
try:
|
2021-03-13 20:37:13 +01:00
|
|
|
with open(filepath, "rt", encoding=encoding) as f:
|
|
|
|
text = f.read()
|
2021-03-08 23:40:03 +01:00
|
|
|
except (OSError, ValueError) as exc:
|
|
|
|
self.set_status_error(f"Failed to open file: {exc}")
|
2021-03-13 20:37:13 +01:00
|
|
|
return
|
|
|
|
self.load_page(Page.from_gemtext(text))
|