Add splashscreen to select providers
This commit is contained in:
parent
969786e299
commit
6010e3a6fa
4 changed files with 121 additions and 45 deletions
|
@ -20,7 +20,10 @@ class SVT(ChannelProvider.ChannelProvider):
|
||||||
m_cachefile = '/tmp/svt_cache'
|
m_cachefile = '/tmp/svt_cache'
|
||||||
|
|
||||||
def __init__(self) -> None:
|
def __init__(self) -> None:
|
||||||
super().__init__('SVT', 'http://www.svtplay.se/rss.xml')
|
rss_url = 'http://www.svtplay.se/rss.xml'
|
||||||
|
logo_url = 'https://upload.wikimedia.org/wikipedia/commons/'
|
||||||
|
logo_url += 'thumb/4/4b/Logotyp_SVT_Play.png/480px-Logotyp_SVT_Play.png'
|
||||||
|
super().__init__('SVT', rss_url, logo_url)
|
||||||
|
|
||||||
if os.path.exists(self.m_cachefile):
|
if os.path.exists(self.m_cachefile):
|
||||||
with open(self.m_cachefile, 'rb') as cachehandle:
|
with open(self.m_cachefile, 'rb') as cachehandle:
|
||||||
|
|
|
@ -18,10 +18,11 @@ class YouTube(ChannelProvider):
|
||||||
m_cachefile = '/tmp/yt_cache'
|
m_cachefile = '/tmp/yt_cache'
|
||||||
|
|
||||||
def __init__(self, channel_id) -> None:
|
def __init__(self, channel_id) -> None:
|
||||||
super().__init__(
|
rss_url = 'https://www.youtube.com/feeds/videos.xml?channel_id={}'.format(
|
||||||
channel_id,
|
channel_id)
|
||||||
'https://www.youtube.com/feeds/videos.xml?channel_id={}'.format(
|
logo_url = 'https://upload.wikimedia.org/wikipedia/commons/thumb'
|
||||||
channel_id))
|
logo_url += '/3/3f/Logo_of_YouTube_(2005-2011).svg/480px-Logo_of_YouTube_(2005-2011).svg.png'
|
||||||
|
super().__init__(channel_id, rss_url, logo_url)
|
||||||
self.m_items: Union[list[Item], None] = None
|
self.m_items: Union[list[Item], None] = None
|
||||||
|
|
||||||
if os.path.exists(self.m_cachefile):
|
if os.path.exists(self.m_cachefile):
|
||||||
|
@ -39,7 +40,8 @@ class YouTube(ChannelProvider):
|
||||||
feed = feedparser.parse(self.get_feed())
|
feed = feedparser.parse(self.get_feed())
|
||||||
|
|
||||||
ydl_opts = {
|
ydl_opts = {
|
||||||
'format': 'worst',
|
'format':
|
||||||
|
'worstvideo[ext=mp4]+worstaudio[ext=m4a]/worstvideo+worstaudio',
|
||||||
'container': 'webm_dash',
|
'container': 'webm_dash',
|
||||||
}
|
}
|
||||||
entries = feed['entries']
|
entries = feed['entries']
|
||||||
|
|
|
@ -2,14 +2,39 @@
|
||||||
|
|
||||||
from typing import Union
|
from typing import Union
|
||||||
|
|
||||||
|
import requests
|
||||||
|
import wx
|
||||||
|
import io
|
||||||
|
|
||||||
from Items import Item
|
from Items import Item
|
||||||
|
|
||||||
|
default_logo = "https://upload.wikimedia.org/wikipedia/commons/"
|
||||||
|
default_logo += "thumb/f/fd/Cartoon_Hand_Playing_Multiple_Online_Videos.svg/"
|
||||||
|
default_logo += "480px-Cartoon_Hand_Playing_Multiple_Online_Videos.svg.png"
|
||||||
|
|
||||||
|
|
||||||
class ChannelProvider:
|
class ChannelProvider:
|
||||||
def __init__(self, provider_name: str, feed: str) -> None:
|
def __init__(self,
|
||||||
|
provider_name: str,
|
||||||
|
feed: str,
|
||||||
|
logo_url: str = default_logo) -> None:
|
||||||
self.m_provider_name = provider_name
|
self.m_provider_name = provider_name
|
||||||
self.m_feed = feed
|
self.m_feed = feed
|
||||||
self.m_items: Union[list[Item], None] = None
|
self.m_items: Union[list[Item], None] = None
|
||||||
|
res = requests.get(logo_url)
|
||||||
|
content = res.content
|
||||||
|
content_bytes = io.BytesIO(content)
|
||||||
|
self.m_logo = wx.Image(content_bytes,
|
||||||
|
type=wx.BITMAP_TYPE_ANY,
|
||||||
|
index=-1)
|
||||||
|
|
||||||
|
def get_logo_as_bitmap(self) -> wx.Bitmap:
|
||||||
|
"""
|
||||||
|
[TODO:description]
|
||||||
|
|
||||||
|
:rtype wx.Image: [TODO:description]
|
||||||
|
"""
|
||||||
|
return wx.Bitmap(self.m_logo)
|
||||||
|
|
||||||
def get_feed(self) -> str:
|
def get_feed(self) -> str:
|
||||||
return self.m_feed
|
return self.m_feed
|
||||||
|
|
84
main.py
84
main.py
|
@ -14,7 +14,6 @@ ChannelProvider = NewType('ChannelProvider', ChannelProvider)
|
||||||
|
|
||||||
|
|
||||||
class Cast(wx.Frame):
|
class Cast(wx.Frame):
|
||||||
|
|
||||||
def __init__(self, *args, **kw):
|
def __init__(self, *args, **kw):
|
||||||
"""__init__.
|
"""__init__.
|
||||||
:param args:
|
:param args:
|
||||||
|
@ -34,17 +33,51 @@ class Cast(wx.Frame):
|
||||||
self.m_control = None
|
self.m_control = None
|
||||||
self.m_panel.SetupScrolling()
|
self.m_panel.SetupScrolling()
|
||||||
self.m_panel.SetSizer(self.m_sizer)
|
self.m_panel.SetSizer(self.m_sizer)
|
||||||
self.m_channels: list[ChannelProvider] = [YouTube.YouTube('UCu6mSoMNzHQiBIOCkHUa2Aw')]#,SVT.SVT()]
|
self.m_providers: list[ChannelProvider] = [
|
||||||
self.show_list(None)
|
YouTube.YouTube('UCu6mSoMNzHQiBIOCkHUa2Aw'),
|
||||||
|
SVT.SVT()
|
||||||
|
]
|
||||||
|
self.m_selected_channel = None
|
||||||
|
self.show_splash(None)
|
||||||
|
|
||||||
def get_chromecasts(self):
|
def get_chromecasts(self) -> None:
|
||||||
|
"""
|
||||||
|
[TODO:description]
|
||||||
|
|
||||||
|
:rtype None: [TODO:description]
|
||||||
|
"""
|
||||||
self.m_chromecasts, self.m_browser = pychromecast.get_chromecasts()
|
self.m_chromecasts, self.m_browser = pychromecast.get_chromecasts()
|
||||||
|
|
||||||
def show_list(self, _):
|
def show_splash(self, _) -> None:
|
||||||
|
self.m_sizer.Clear(delete_windows=True)
|
||||||
|
self.m_sizer = wx.BoxSizer(wx.VERTICAL)
|
||||||
|
channel_index = 0
|
||||||
|
for provider in self.m_providers:
|
||||||
|
bitmap = provider.get_logo_as_bitmap()
|
||||||
|
btn = wx.BitmapButton(self.m_panel,
|
||||||
|
id=channel_index,
|
||||||
|
bitmap=bitmap)
|
||||||
|
btn.Bind(wx.EVT_BUTTON,
|
||||||
|
lambda event, index=channel_index: self.show_list(
|
||||||
|
event, index))
|
||||||
|
self.m_sizer.Add(btn)
|
||||||
|
channel_index += 1
|
||||||
|
|
||||||
|
self.m_panel.SetSizer(self.m_sizer)
|
||||||
|
self.m_sizer.Fit(self)
|
||||||
|
self.m_panel.Layout()
|
||||||
|
|
||||||
|
def show_list(self, _, index=0) -> None:
|
||||||
|
"""
|
||||||
|
Shows a list of videos
|
||||||
|
|
||||||
|
:param _ [TODO:type]: [TODO:description]
|
||||||
|
:rtype None: [TODO:description]
|
||||||
|
"""
|
||||||
self.m_sizer.Clear(delete_windows=True)
|
self.m_sizer.Clear(delete_windows=True)
|
||||||
self.m_sizer = wx.BoxSizer(wx.VERTICAL)
|
self.m_sizer = wx.BoxSizer(wx.VERTICAL)
|
||||||
|
|
||||||
for channel in self.m_channels:
|
channel = self.m_providers[index]
|
||||||
if channel.wait():
|
if channel.wait():
|
||||||
with wx.BusyInfo("Please wait, working..."):
|
with wx.BusyInfo("Please wait, working..."):
|
||||||
index = 0
|
index = 0
|
||||||
|
@ -52,21 +85,22 @@ class Cast(wx.Frame):
|
||||||
time.sleep(1)
|
time.sleep(1)
|
||||||
wx.GetApp().Yield()
|
wx.GetApp().Yield()
|
||||||
|
|
||||||
|
backbtn = wx.Button(self.m_panel, -1, label="Go back", size=(480, 40))
|
||||||
|
backbtn.Bind(wx.EVT_BUTTON, lambda event: self.show_splash(event))
|
||||||
|
self.m_sizer.Add(backbtn, wx.ALIGN_CENTER_VERTICAL)
|
||||||
|
self.m_sizer.AddSpacer(10)
|
||||||
for item in channel.get_items():
|
for item in channel.get_items():
|
||||||
title = wx.StaticText(self.m_panel, -1)
|
title = wx.StaticText(self.m_panel, -1)
|
||||||
title.SetLabelMarkup("<span weight='bold' >{}</span>".format(item['title']))
|
title.SetLabelMarkup("<span weight='bold' >{}</span>".format(
|
||||||
description = wx.StaticText(self.m_panel, -1,
|
item['title']))
|
||||||
item['description'])
|
description = wx.StaticText(self.m_panel, -1, item['description'])
|
||||||
|
|
||||||
description.Wrap(478)
|
description.Wrap(478)
|
||||||
bitmap = item['thumbnail']
|
bitmap = item['thumbnail']
|
||||||
btn = wx.BitmapButton(self.m_panel,
|
btn = wx.BitmapButton(self.m_panel, id=self.m_index, bitmap=bitmap)
|
||||||
id=self.m_index,
|
|
||||||
bitmap=bitmap)
|
|
||||||
btn.Bind(wx.EVT_BUTTON,
|
btn.Bind(wx.EVT_BUTTON,
|
||||||
lambda event, link=item['link']: self.show_player(
|
lambda event, link=item['link'], provider_index=index:
|
||||||
event, link))
|
self.show_player(event, link, provider_index))
|
||||||
self.m_sizer.Add(title)
|
self.m_sizer.Add(title)
|
||||||
self.m_sizer.Add(btn)
|
self.m_sizer.Add(btn)
|
||||||
self.m_sizer.Add(description)
|
self.m_sizer.Add(description)
|
||||||
|
@ -76,7 +110,13 @@ class Cast(wx.Frame):
|
||||||
self.m_sizer.Fit(self)
|
self.m_sizer.Fit(self)
|
||||||
self.m_panel.Layout()
|
self.m_panel.Layout()
|
||||||
|
|
||||||
def show_player(self, _, uri):
|
def show_player(self, _, uri, provider_index: int):
|
||||||
|
"""
|
||||||
|
Show the video player
|
||||||
|
|
||||||
|
:param _ event: unused
|
||||||
|
:param uri str: the link to the video stream
|
||||||
|
"""
|
||||||
self.m_sizer.Clear(delete_windows=True)
|
self.m_sizer.Clear(delete_windows=True)
|
||||||
self.m_sizer = wx.GridBagSizer()
|
self.m_sizer = wx.GridBagSizer()
|
||||||
self.m_control = wx.media.MediaCtrl(
|
self.m_control = wx.media.MediaCtrl(
|
||||||
|
@ -89,7 +129,9 @@ class Cast(wx.Frame):
|
||||||
pause_button = wx.Button(self.m_panel, -1, "Pause")
|
pause_button = wx.Button(self.m_panel, -1, "Pause")
|
||||||
|
|
||||||
back_button = wx.Button(self.m_panel, -1, "Back")
|
back_button = wx.Button(self.m_panel, -1, "Back")
|
||||||
back_button.Bind(wx.EVT_BUTTON, self.show_list)
|
back_button.Bind(
|
||||||
|
wx.EVT_BUTTON,
|
||||||
|
lambda event, index=provider_index: self.show_list(event, index))
|
||||||
|
|
||||||
self.m_sizer.Add(self.m_control, (0, 0))
|
self.m_sizer.Add(self.m_control, (0, 0))
|
||||||
self.m_sizer.SetItemSpan(0, (0, 6))
|
self.m_sizer.SetItemSpan(0, (0, 6))
|
||||||
|
@ -97,7 +139,8 @@ class Cast(wx.Frame):
|
||||||
self.m_sizer.Add(pause_button, (1, 2))
|
self.m_sizer.Add(pause_button, (1, 2))
|
||||||
self.m_sizer.Add(back_button, (1, 3))
|
self.m_sizer.Add(back_button, (1, 3))
|
||||||
|
|
||||||
if not self.m_chromecast_thr.is_alive() and not self.m_selected_chromecast:
|
if not self.m_chromecast_thr.is_alive(
|
||||||
|
) and not self.m_selected_chromecast:
|
||||||
chromecast_button = wx.Button(self.m_panel, -1, "Cast")
|
chromecast_button = wx.Button(self.m_panel, -1, "Cast")
|
||||||
chromecast_button.Bind(
|
chromecast_button.Bind(
|
||||||
wx.EVT_BUTTON,
|
wx.EVT_BUTTON,
|
||||||
|
@ -127,7 +170,10 @@ class Cast(wx.Frame):
|
||||||
|
|
||||||
for cast in self.m_chromecasts:
|
for cast in self.m_chromecasts:
|
||||||
friendly_name = cast.cast_info.friendly_name
|
friendly_name = cast.cast_info.friendly_name
|
||||||
btn = wx.Button(self.m_panel, id=-1, label=friendly_name, size=(480,40))
|
btn = wx.Button(self.m_panel,
|
||||||
|
id=-1,
|
||||||
|
label=friendly_name,
|
||||||
|
size=(480, 40))
|
||||||
btn.Bind(wx.EVT_BUTTON,
|
btn.Bind(wx.EVT_BUTTON,
|
||||||
lambda event, chromecast=cast, muri=uri: self.
|
lambda event, chromecast=cast, muri=uri: self.
|
||||||
set_chromecast(event, chromecast, muri))
|
set_chromecast(event, chromecast, muri))
|
||||||
|
|
Loading…
Add table
Reference in a new issue