2020-01-26 11:13:45 +01:00
|
|
|
from html.parser import HTMLParser
|
2022-11-21 08:31:09 +01:00
|
|
|
import os
|
2019-08-24 14:14:46 +02:00
|
|
|
import re
|
2019-08-29 11:47:44 +02:00
|
|
|
import shutil
|
|
|
|
import subprocess
|
2019-08-24 14:14:46 +02:00
|
|
|
|
2020-04-15 14:01:23 +02:00
|
|
|
from datetime import datetime, timezone
|
2019-08-27 14:34:51 +02:00
|
|
|
|
2019-08-24 14:14:46 +02:00
|
|
|
HASHTAG_PATTERN = re.compile(r'(?<!\w)(#\w+)\b')
|
|
|
|
|
|
|
|
|
2019-08-27 14:34:51 +02:00
|
|
|
def parse_datetime(value):
|
|
|
|
"""Returns an aware datetime in local timezone"""
|
2019-09-04 08:36:12 +02:00
|
|
|
|
|
|
|
# In Python < 3.7, `%z` does not match `Z` offset
|
|
|
|
# https://docs.python.org/3.7/library/datetime.html#strftime-and-strptime-behavior
|
|
|
|
if value.endswith("Z"):
|
2020-04-15 14:01:23 +02:00
|
|
|
dttm = datetime.strptime(value, "%Y-%m-%dT%H:%M:%S.%fZ").replace(tzinfo=timezone.utc)
|
2019-09-04 08:36:12 +02:00
|
|
|
else:
|
|
|
|
dttm = datetime.strptime(value, "%Y-%m-%dT%H:%M:%S.%f%z")
|
|
|
|
|
2022-11-21 08:31:09 +01:00
|
|
|
# When running tests return datetime in UTC so that tests don't depend on
|
|
|
|
# the local timezone
|
|
|
|
if "PYTEST_CURRENT_TEST" in os.environ:
|
|
|
|
return dttm.astimezone(timezone.utc)
|
|
|
|
|
2019-09-04 08:36:12 +02:00
|
|
|
return dttm.astimezone()
|
2019-08-28 15:32:57 +02:00
|
|
|
|
|
|
|
|
|
|
|
def highlight_keys(text, high_attr, low_attr=""):
|
|
|
|
"""
|
|
|
|
Takes a string and adds high_attr attribute to parts in square brackets,
|
|
|
|
and optionally low_attr attribute to parts outside square brackets.
|
|
|
|
|
|
|
|
The result can be rendered using a urwid.Text widget.
|
|
|
|
|
|
|
|
For example:
|
|
|
|
|
|
|
|
>>> highlight_keys("[P]rint [V]iew", "blue")
|
|
|
|
>>> [('blue', 'P'), 'rint ', ('blue', 'V'), 'iew']
|
|
|
|
"""
|
|
|
|
def _gen():
|
|
|
|
highlighted = False
|
|
|
|
for part in re.split("\\[|\\]", text):
|
|
|
|
if part:
|
|
|
|
if highlighted:
|
|
|
|
yield (high_attr, part) if high_attr else part
|
|
|
|
else:
|
|
|
|
yield (low_attr, part) if low_attr else part
|
|
|
|
highlighted = not highlighted
|
|
|
|
return list(_gen())
|
|
|
|
|
|
|
|
|
2022-12-20 22:28:24 +01:00
|
|
|
def highlight_hashtags(line, followed_tags, attr="hashtag",\
|
|
|
|
followed_attr="followed_hashtag"):
|
|
|
|
hline = []
|
|
|
|
for p in re.split(HASHTAG_PATTERN, line):
|
|
|
|
if p.startswith("#"):
|
|
|
|
if p[1:].lower() in (t.lower() for t in followed_tags):
|
|
|
|
hline.append((followed_attr,p))
|
|
|
|
else:
|
|
|
|
hline.append((attr,p))
|
|
|
|
else:
|
|
|
|
hline.append(p)
|
|
|
|
return hline
|
2019-08-29 11:47:44 +02:00
|
|
|
|
|
|
|
|
|
|
|
def show_media(paths):
|
|
|
|
"""
|
|
|
|
Attempt to open an image viewer to show given media files.
|
|
|
|
|
|
|
|
FIXME: This is not very thought out, but works for me.
|
|
|
|
Once settings are implemented, add an option for the user to configure their
|
|
|
|
prefered media viewer.
|
|
|
|
"""
|
|
|
|
viewer = None
|
|
|
|
potential_viewers = [
|
|
|
|
"feh",
|
|
|
|
"eog",
|
|
|
|
"display"
|
|
|
|
]
|
|
|
|
for v in potential_viewers:
|
|
|
|
viewer = shutil.which(v)
|
|
|
|
if viewer:
|
|
|
|
break
|
|
|
|
|
|
|
|
if not viewer:
|
|
|
|
raise Exception("Cannot find an image viewer")
|
|
|
|
|
|
|
|
subprocess.run([viewer] + paths)
|
2020-01-26 11:13:45 +01:00
|
|
|
|
|
|
|
|
|
|
|
class LinkParser(HTMLParser):
|
|
|
|
|
|
|
|
def reset(self):
|
|
|
|
super().reset()
|
|
|
|
self.links = []
|
|
|
|
|
|
|
|
def handle_starttag(self, tag, attrs):
|
|
|
|
if tag == "a":
|
|
|
|
href, title = None, None
|
|
|
|
for name, value in attrs:
|
|
|
|
if name == "href":
|
|
|
|
href = value
|
|
|
|
if name == "title":
|
|
|
|
title = value
|
|
|
|
if href:
|
|
|
|
self.links.append((href, title))
|
|
|
|
|
|
|
|
|
|
|
|
def parse_content_links(content):
|
|
|
|
"""Parse <a> tags from status's `content` and return them as a list of
|
|
|
|
(href, title), where `title` may be None.
|
|
|
|
"""
|
|
|
|
parser = LinkParser()
|
|
|
|
parser.feed(content)
|
|
|
|
return parser.links[:]
|