main_screen.py 10 KB

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