main_screen.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. import hashlib
  2. import json
  3. import logging
  4. import os
  5. import time
  6. import urllib
  7. import urllib2
  8. from threading import Thread
  9. import pygame
  10. from .screen_objects import (Progressbar, ScreenObjectsManager, TextItem,
  11. TouchAndTextItem)
  12. from .input_manager import InputManager
  13. logger = logging.getLogger(__name__)
  14. class MainScreen():
  15. def __init__(self, size, manager, cache, core, fonts):
  16. self.core = core
  17. self.size = size
  18. self.base_size = self.size[1] / 8
  19. self.fonts = fonts
  20. self.manager = manager
  21. self.track = None
  22. self.cache = cache
  23. self.image = None
  24. self.artists = None
  25. self.image_now_loaded = False
  26. self.touch_text_manager = ScreenObjectsManager()
  27. current_track = self.core.playback.current_track.get()
  28. if current_track is None:
  29. self.track_playback_ended(None, None)
  30. else:
  31. self.track_started(current_track)
  32. def update(self, screen, update_all):
  33. if self.track is not None:
  34. self.touch_text_manager.get_touch_object(
  35. "time_progress").set_value(
  36. self.core.playback.time_position.get() / 1000)
  37. self.touch_text_manager.get_touch_object("time_progress").set_text(
  38. time.strftime('%M:%S', time.gmtime(
  39. self.core.playback.time_position.get() / 1000)) + "/" +
  40. time.strftime('%M:%S', time.gmtime(
  41. self.track.length / 1000)))
  42. if update_all:
  43. if self.image_now_loaded:
  44. if self.image is not None:
  45. screen.blit(self.image, (self.base_size / 2, self.base_size + self.base_size / 2))
  46. self.image_now_loaded = False
  47. self.touch_text_manager.render(screen)
  48. return screen
  49. def track_started(self, track):
  50. self.image = None
  51. x = self.base_size * 5
  52. width = self.size[0] - self.base_size / 2 - x
  53. # Load all artists
  54. self.artists = []
  55. for artist in track.artists:
  56. self.artists.append(artist)
  57. # Track name
  58. label = TextItem(self.fonts['base'], MainScreen.get_track_name(track),
  59. (x, self.base_size * 2),
  60. (width, self.size[1]))
  61. self.touch_text_manager.set_object("track_name", label)
  62. # Album name
  63. label = TextItem(self.fonts['base'],
  64. MainScreen.get_track_album_name(track),
  65. (x, self.base_size * 3),
  66. (width, self.size[1]))
  67. self.touch_text_manager.set_object("album_name", label)
  68. # Artist
  69. label = TextItem(self.fonts['base'], self.get_artist_string(),
  70. (x, self.base_size * 4), (width, self.size[1]))
  71. self.touch_text_manager.set_object("artist_name", label)
  72. # Previous track button
  73. button = TouchAndTextItem(self.fonts['icon'], u"\ue61c",
  74. (0, self.base_size * 6), None)
  75. self.touch_text_manager.set_touch_object("previous", button)
  76. size_1 = button.get_right_pos()
  77. size_2 = self.fonts['icon'].size(u"\ue61d")[0]
  78. button = TouchAndTextItem(self.fonts['icon'], u"\ue61d",
  79. (self.size[0] - size_2, self.base_size * 6),
  80. None)
  81. self.touch_text_manager.set_touch_object("next", button)
  82. # Progress
  83. progress = Progressbar(self.fonts['base'],
  84. time.strftime('%M:%S', time.gmtime(
  85. 0)) + "/" + time.strftime('%M:%S',
  86. time.gmtime(0)),
  87. (size_1, self.base_size * 6),
  88. (
  89. self.size[0] - size_1 - size_2, self.base_size),
  90. track.length / 1000, False)
  91. self.touch_text_manager.set_touch_object("time_progress", progress)
  92. self.track = track
  93. if not self.is_image_in_cache():
  94. thread = Thread(target=self.download_image(0))
  95. thread.start()
  96. else:
  97. self.load_image()
  98. def get_dirty_area(self):
  99. return self.touch_text_manager.get_dirty_area()
  100. def get_artist_string(self):
  101. artists_string = ''
  102. for artist in self.artists:
  103. artists_string += artist.name + ', '
  104. if len(artists_string) > 2:
  105. artists_string = artists_string[:-2]
  106. elif len(artists_string) == 0:
  107. artists_string = "Unknow Artist"
  108. return artists_string
  109. def get_image_file_name(self):
  110. name = MainScreen.get_track_album_name(
  111. self.track) + '-' + self.get_artist_string()
  112. md5name = hashlib.md5(name.encode('utf-8')).hexdigest()
  113. return md5name
  114. def get_cover_folder(self):
  115. if not os.path.isdir(self.cache + "/covers"):
  116. os.makedirs(self.cache + "/covers")
  117. return self.cache + "/covers/"
  118. def is_image_in_cache(self):
  119. self.get_cover_folder()
  120. return os.path.isfile(
  121. self.get_cover_folder() + self.get_image_file_name())
  122. def download_image(self, artist_index):
  123. if artist_index < len(self.artists):
  124. try:
  125. safe_artist = urllib.quote_plus(
  126. self.artists[artist_index].name)
  127. safe_album = urllib.quote_plus(
  128. MainScreen.get_track_album_name(self.track))
  129. url = "http://ws.audioscrobbler.com/2.0/?"
  130. params = "method=album.getinfo&" + \
  131. "api_key=59a04c6a73fb99d6e8996e01db306829&" \
  132. + "artist=" \
  133. + safe_artist + "&album=" + safe_album + \
  134. "&format=json"
  135. response = urllib2.urlopen(url + params)
  136. data = json.load(response)
  137. image = data['album']['image'][-1]['#text']
  138. urllib.urlretrieve(image,
  139. self.get_cover_folder() +
  140. self.get_image_file_name())
  141. self.load_image()
  142. except:
  143. self.download_image(artist_index + 1)
  144. else:
  145. logger.info("Cover could not be downloaded")
  146. # There is no cover so it will use all the screen size for the text
  147. width = self.size[0] - self.base_size
  148. current = TextItem(self.fonts['base'],
  149. MainScreen.get_track_name(self.track),
  150. (self.base_size / 2, self.base_size * 2),
  151. (width, -1))
  152. self.touch_text_manager.set_object("track_name", current)
  153. current = TextItem(self.fonts['base'],
  154. MainScreen.get_track_album_name(self.track),
  155. (self.base_size / 2, self.base_size * 3),
  156. (width, -1))
  157. self.touch_text_manager.set_object("album_name", current)
  158. current = TextItem(self.fonts['base'], self.get_artist_string(),
  159. (self.base_size / 2, self.base_size * 4),
  160. (width, -1))
  161. self.touch_text_manager.set_object("artist_name", current)
  162. def track_playback_ended(self, tl_track, time_position):
  163. self.image = None
  164. # There is no cover so it will use all the screen size for the text
  165. width = self.size[0] - self.base_size
  166. current = TextItem(self.fonts['base'], "Stopped",
  167. (self.base_size / 2, self.base_size * 2),
  168. (width, -1))
  169. self.touch_text_manager.set_object("track_name", current)
  170. current = TextItem(self.fonts['base'], "",
  171. (self.base_size / 2, self.base_size * 3),
  172. (width, -1))
  173. self.touch_text_manager.set_object("album_name", current)
  174. current = TextItem(self.fonts['base'], "",
  175. (self.base_size / 2, self.base_size * 4),
  176. (width, -1))
  177. self.touch_text_manager.set_object("artist_name", current)
  178. def load_image(self):
  179. size = self.base_size * 4
  180. self.image = pygame.transform.scale(
  181. pygame.image.load(
  182. self.get_cover_folder() +
  183. self.get_image_file_name()).convert(),
  184. (size, size))
  185. self.image_now_loaded = True
  186. def touch_event(self, event):
  187. if event.type == InputManager.click:
  188. objects = self.touch_text_manager.get_touch_objects_in_pos(
  189. event.current_pos)
  190. if objects is not None:
  191. for key in objects:
  192. if key == "time_progress":
  193. value = self.touch_text_manager.get_touch_object(
  194. key).get_pos_value(event.current_pos) * 1000
  195. self.core.playback.seek(value)
  196. elif key == "previous":
  197. self.core.playback.previous()
  198. elif key == "next":
  199. self.core.playback.next()
  200. elif event.type == InputManager.swipe:
  201. if event.direction == InputManager.left:
  202. self.core.playback.next()
  203. elif event.direction == InputManager.right:
  204. self.core.playback.previous()
  205. elif event.direction == InputManager.up:
  206. volume = self.core.playback.volume.get() + 10
  207. if volume > 100:
  208. volume = 100
  209. self.manager.backend.tell(
  210. {'action': 'volume', 'value': volume})
  211. self.manager.volume_changed(volume)
  212. elif event.direction == InputManager.down:
  213. volume = self.core.playback.volume.get() - 10
  214. if volume < 0:
  215. volume = 0
  216. self.manager.backend.tell(
  217. {'action': 'volume', 'value': volume})
  218. self.manager.volume_changed(volume)
  219. @staticmethod
  220. def get_track_name(track):
  221. if track.name is None:
  222. return track.uri
  223. else:
  224. return track.name
  225. @staticmethod
  226. def get_track_album_name(track):
  227. if track.album is not None and track.album.name is not None and len(
  228. track.album.name) > 0:
  229. return track.album.name
  230. else:
  231. return "Unknow Album"