scrobble_list.html 43 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750
  1. {% extends "base.html" %}
  2. {% load humanize %}
  3. {% load static %}
  4. {% block head_extra %}
  5. <style>
  6. .container { margin-bottom:100px; }
  7. h2 { padding-top:20px; }
  8. .image-wrapper {
  9. contain: content;
  10. }
  11. .image-wrapper :hover {
  12. background:rgba(0,0,0,0.3);
  13. }
  14. .caption {
  15. position: fixed;
  16. top: 5px;
  17. left: 5px;
  18. padding: 3px;
  19. font-size: 90%;
  20. color:white;
  21. background:rgba(0,0,0,0.4);
  22. }
  23. .caption-medium {
  24. position: fixed;
  25. top: 5px;
  26. left: 5px;
  27. padding: 3px;
  28. font-size: 75%;
  29. color:white;
  30. background:rgba(0,0,0,0.4);
  31. }
  32. .caption-small {
  33. position: fixed;
  34. top: 5px;
  35. left: 5px;
  36. padding: 3px;
  37. font-size: 60%;
  38. color:white;
  39. background:rgba(0,0,0,0.4);
  40. }
  41. </style>
  42. {% endblock %}
  43. {% block content %}
  44. <main class="col-md-9 ms-sm-auto col-lg-10 px-md-4">
  45. <div
  46. class="d-flex justify-content-between flex-wrap flex-md-nowrap align-items-center pt-3 pb-2 mb-3 border-bottom">
  47. <h1 class="h2">Dashboard</h1>
  48. <div class="btn-toolbar mb-2 mb-md-0">
  49. {% if user.is_authenticated %}
  50. <div class="btn-group me-2">
  51. {% if user.profile.lastfm_username and not user.profile.lastfm_auto_import %}
  52. <form action="{% url 'scrobbles:lastfm-import' %}" method="get">
  53. <button type="submit" class="btn btn-sm btn-outline-secondary">Last.fm Sync</button>
  54. </form>
  55. {% endif %}
  56. <button type="button" class="btn btn-sm btn-outline-secondary" data-bs-toggle="modal"
  57. data-bs-target="#importModal">Import</button>
  58. <button type="button" class="btn btn-sm btn-outline-secondary" data-bs-toggle="modal"
  59. data-bs-target="#exportModal">Export</button>
  60. </div>
  61. {% endif %}
  62. <div class="dropdown">
  63. <button type="button" class="btn btn-sm btn-outline-secondary dropdown-toggle" id="graphDateButton"
  64. data-bs-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
  65. <div data-feather="calendar"></div>
  66. This week
  67. </button>
  68. <div class="dropdown-menu" data-bs-toggle="#graphDataChange" aria-labelledby="graphDateButton">
  69. <a class="dropdown-item" href="#">This month</a>
  70. <a class="dropdown-item" href="#">This year</a>
  71. </div>
  72. </div>
  73. </div>
  74. </div>
  75. {% if not user.is_authenticated %}
  76. <p>Today <b>{{counts.today}}</b> | This Week <b>{{counts.week}}</b> | This Month <b>{{counts.month}}</b> | This Year <b>{{counts.year}}</b> | All Time <b>{{counts.alltime}}</b></p>
  77. <canvas class="my-4 w-100" id="myChart" width="900" height="300"></canvas>
  78. {% endif %}
  79. <div class="container">
  80. {% if user.is_authenticated %}
  81. <div class="row">
  82. <h2>Top Artist</h2>
  83. <ul class="nav nav-tabs" id="artistTab" role="tablist">
  84. {% for key, name in chart_keys.items %}
  85. <li class="nav-item" role="presentation">
  86. <button class="nav-link {% if forloop.counter == 2 %}active{% endif %}"
  87. id="artist-{{key}}-tab" data-bs-toggle="tab" data-bs-target="#artist-{{key}}"
  88. type="button" role="tab" aria-controls="home" aria-selected="true">{{name}}</button>
  89. </li>
  90. {% endfor %}
  91. </ul>
  92. <div class="tab-content" id="artistTabContent" class="maloja-chart">
  93. {% for key, artists in current_artist_charts.items %}
  94. <div class="tab-pane fade {% if forloop.counter == 2 %}show active{% endif %}" id="artist-{{key}}" role="tabpanel" aria-labelledby="artist-{{key}}-tab">
  95. <div style="display:block">
  96. <div style="float:left;">
  97. <div class="image-wrapper" style="display:flex; flex-wrap: wrap; margin:0">
  98. <div class="caption">#1 {{artists.0.name}}</div>
  99. {% if artists.0 %}
  100. {% if artists.0.thumbnail %}
  101. <a href="{{artists.0.get_absolute_url}}"><img lt="{{artists.0.name}}" src="{{artists.0.thumbnail.url}}" width="300px"></a>
  102. {% else %}
  103. <a href="{{artists.0.get_absolute_url}}"><img lt="{{artists.0.name}}" src="{% static "images/not-found.jpg" %}" width="300px"></a>
  104. {% endif %}
  105. {% endif %}
  106. </div>
  107. </div>
  108. <div style="float:left; width:300px;">
  109. <div style="display:flex; flex-wrap: wrap;">
  110. <div class="image-wrapper" class="image-wrapper" style="width:50%">
  111. <div class="caption-medium">#2 {{artists.1.name}}</div>
  112. {% if artists.1 %}
  113. {% if artists.1.thumbnail %}
  114. <a href="{{artists.1.get_absolute_url}}"><img lt="{{artists.1.name}}" src="{{artists.1.thumbnail.url}}" width="150px"></a>
  115. {% else %}
  116. <a href="{{artists.1.get_absolute_url}}"><img src="{% static "images/not-found.jpg" %}" width="150px"></a>
  117. {% endif %}
  118. {% endif %}
  119. </div>
  120. <div class="image-wrapper" class="image-wrapper" style="width:50%">
  121. <div class="caption-medium">#3 {{artists.2.name}}</div>
  122. {% if artists.2 %}
  123. {% if artists.2.thumbnail %}
  124. <a href="{{artists.2.get_absolute_url}}"><img src="{{artists.2.thumbnail.url}}" width="150px"></a>
  125. {% else %}
  126. <a href="{{artists.2.get_absolute_url}}"><img src="{% static "images/not-found.jpg" %}" width="150px"></a>
  127. {% endif %}
  128. {% endif %}
  129. </div>
  130. <div class="image-wrapper" class="image-wrapper" style="width:50%">
  131. <div class="caption-medium">#4 {{artists.3.name}}</div>
  132. {% if artists.3 %}
  133. {% if artists.3.thumbnail %}
  134. <a href="{{artists.3.get_absolute_url}}"><img src="{{artists.3.thumbnail.url}}" width="150px"></a>
  135. {% else %}
  136. <a href="{{artists.3.get_absolute_url}}"><img src="{% static "images/not-found.jpg" %}" width="150px"></a>
  137. {% endif %}
  138. {% endif %}
  139. </div>
  140. <div class="image-wrapper" class="image-wrapper" style="width:50%">
  141. <div class="caption-medium">#5 {{artists.4.name}}</div>
  142. {% if artists.4 %}
  143. {% if artists.4.thumbnail %}
  144. <a href="{{artists.4.get_absolute_url}}"><img src="{{artists.4.thumbnail.url}}" width="150px"></a>
  145. {% else %}
  146. <a href="{{artists.4.get_absolute_url}}"><img src="{% static "images/not-found.jpg" %}" width="150px"></a>
  147. {% endif %}
  148. {% endif %}
  149. </div>
  150. </div>
  151. </div>
  152. <div style="float:left; width:300px;">
  153. <div style="display:flex; flex-wrap: wrap;">
  154. <div class="image-wrapper" class="image-wrapper" class="image-wrapper" style="width:33;">
  155. <div class="caption-small">#6 {{artists.5.name}}</div>
  156. {% if artists.5 %}
  157. {% if artists.5.thumbnail %}
  158. <a href="{{artists.5.get_absolute_url}}"><img src="{{artists.5.thumbnail.url}}" width="100px"></a>
  159. {% else %}
  160. <a href="{{artists.5.get_absolute_url}}"><img src="{% static "images/not-found.jpg" %}" width="100px"></a>
  161. {% endif %}
  162. {% endif %}
  163. </div>
  164. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  165. <div class="caption-small">#7 {{artists.6.name}}</div>
  166. {% if artists.6 %}
  167. {% if artists.6.thumbnail %}
  168. <a href="{{artists.6.get_absolute_url}}"><img src="{{artists.6.thumbnail.url}}" width="100px"></a>
  169. {% else %}
  170. <a href="{{artists.6.get_absolute_url}}"><img src="{% static "images/not-found.jpg" %}" width="100px"></a>
  171. {% endif %}
  172. {% endif %}
  173. </div>
  174. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  175. <div class="caption-small">#8 {{artists.7.name}}</div>
  176. {% if artists.7 %}
  177. {% if artists.7.thumbnail %}
  178. <a href="{{artists.7.get_absolute_url}}"><img src="{{artists.7.thumbnail.url}}" width="100px"></a>
  179. {% else %}
  180. <a href="{{artists.7.get_absolute_url}}"><img src="{% static "images/not-found.jpg" %}" width="100px"></a>
  181. {% endif %}
  182. {% endif %}
  183. </div>
  184. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  185. <div class="caption-small">#9 {{artists.8.name}}</div>
  186. {% if artists.8 %}
  187. {% if artists.8.thumbnail %}
  188. <a href="{{artists.8.get_absolute_url}}"><img src="{{artists.8.thumbnail.url}}" width="100px"></a>
  189. {% else %}
  190. <a href="{{artists.8.get_absolute_url}}"><img src="{% static "images/not-found.jpg" %}" width="100px"></a>
  191. {% endif %}
  192. {% endif %}
  193. </div>
  194. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  195. <div class="caption-small">#10 {{artists.9.name}}</div>
  196. {% if artists.9 %}
  197. {% if artists.9.thumbnail %}
  198. <a href="{{artists.9.get_absolute_url}}"><img src="{{artists.9.thumbnail.url}}" width="100px"></a>
  199. {% else %}
  200. <a href="{{artists.9.get_absolute_url}}"><img src="{% static "images/not-found.jpg" %}" width="100px"></a>
  201. {% endif %}
  202. {% endif %}
  203. </div>
  204. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  205. <div class="caption-small">#11 {{artists.10.name}}</div>
  206. {% if artists.10 %}
  207. {% if artists.10.thumbnail %}
  208. <a href="{{artists.10.get_absolute_url}}"><img src="{{artists.10.thumbnail.url}}" width="100px"></a>
  209. {% else %}
  210. <a href="{{artists.10.get_absolute_url}}"><img src="{% static "images/not-found.jpg" %}" width="100px"></a>
  211. {% endif %}
  212. {% endif %}
  213. </div>
  214. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  215. <div class="caption-small">#12 {{artists.11.name}}</div>
  216. {% if artists.11 %}
  217. {% if artists.11.thumbnail %}
  218. <a href="{{artists.11.get_absolute_url}}"><img src="{{artists.11.thumbnail.url}}" width="100px"></a>
  219. {% else %}
  220. <a href="{{artists.11.get_absolute_url}}"><img src="{% static "images/not-found.jpg" %}" width="100px"></a>
  221. {% endif %}
  222. {% endif %}
  223. </div>
  224. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  225. <div class="caption-small">#13 {{artists.12.name}}</div>
  226. {% if artists.12 %}
  227. {% if artists.12.thumbnail %}
  228. <a href="{{artists.12.get_absolute_url}}"><img src="{{artists.12.thumbnail.url}}" width="100px"></a>
  229. {% else %}
  230. <a href="{{artists.12.get_absolute_url}}"><img src="{% static "images/not-found.jpg" %}" width="100px"></a>
  231. {% endif %}
  232. {% endif %}
  233. </div>
  234. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  235. <div class="caption-small">#14 {{artists.13.name}}</div>
  236. {% if artists.13 %}
  237. {% if artists.13.thumbnail %}
  238. <a href="{{artists.13.get_absolute_url}}"><img src="{{artists.13.thumbnail.url}}" width="100px"></a>
  239. {% else %}
  240. <a href="{{artists.13.get_absolute_url}}"><img src="{% static "images/not-found.jpg" %}" width="100px"></a>
  241. {% endif %}
  242. {% endif %}
  243. </div>
  244. </div>
  245. </div>
  246. </div>
  247. </div>
  248. {% endfor %}
  249. </div>
  250. </div>
  251. <div class="row">
  252. <h2>Top Tracks</h2>
  253. <ul class="nav nav-tabs" id="trackTab" role="tablist">
  254. {% for key, name in chart_keys.items %}
  255. <li class="nav-item" role="presentation">
  256. <button class="nav-link {% if forloop.counter == 2 %}active{% endif %}" id="track-{{key}}-tab" data-bs-toggle="tab"
  257. data-bs-target="#track-{{key}}" type="button" role="tab" aria-controls="home" aria-selected="true">{{name}}</button>
  258. </li>
  259. {% endfor %}
  260. </ul>
  261. <div class="tab-content" id="trackTabContent" class="maloja-chart">
  262. {% for chart_name, tracks in current_track_charts.items %}
  263. <div class="tab-pane fade {% if forloop.counter == 2 %}show active{% endif %}" id="track-{{chart_name}}" role="tabpanel" aria-labelledby="track-{{chart_name}}-tab">
  264. <div style="display:block">
  265. <div style="float:left;">
  266. <div class="image-wrapper" style="display:flex; flex-wrap: wrap; margin:0">
  267. <div class="caption">#1 {{tracks.0.title}}</div>
  268. {% if tracks.0 %}
  269. {% if tracks.0.album.cover_image %}
  270. <a href="{{tracks.0.get_absolute_url}}"><img src="{{tracks.0.album.cover_image.url}}" width="300px"></a>
  271. {% else %}
  272. <a href="{{tracks.0.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="300px"></a>
  273. {% endif %}
  274. {% endif %}
  275. </div>
  276. </div>
  277. <div style="float:left; width:300px;">
  278. <div style="display:flex; flex-wrap: wrap;">
  279. <div class="image-wrapper" style="width:50%">
  280. <div class="caption-medium">#2 {{tracks.1.title}}</div>
  281. {% if tracks.1 %}
  282. {% if tracks.1.album.cover_image %}
  283. <a href="{{tracks.1.get_absolute_url}}"><img src="{{tracks.1.album.cover_image.url}}" width="150px"></a>
  284. {% else %}
  285. <a href="{{tracks.1.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="150px"></a>
  286. {% endif %}
  287. {% endif %}
  288. </div>
  289. <div class="image-wrapper" style="width:50%">
  290. <div class="caption-medium">#3 {{tracks.2.title}}</div>
  291. {% if tracks.2 %}
  292. {% if tracks.2.album.cover_image %}
  293. <a href="{{tracks.2.get_absolute_url}}"><img src="{{tracks.2.album.cover_image.url}}" width="150px"></a>
  294. {% else %}
  295. <a href="{{tracks.2.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="150px"></a>
  296. {% endif %}
  297. {% endif %}
  298. </div>
  299. <div class="image-wrapper" style="width:50%">
  300. <div class="caption-medium">#4 {{tracks.3.title}}</div>
  301. {% if tracks.3 %}
  302. {% if tracks.3.album.cover_image %}
  303. <a href="{{tracks.3.get_absolute_url}}"><img src="{{tracks.3.album.cover_image.url}}" width="150px"></a>
  304. {% else %}
  305. <a href="{{tracks.3.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="150px"></a>
  306. {% endif %}
  307. {% endif %}
  308. </div>
  309. <div class="image-wrapper" style="width:50%">
  310. <div class="caption-medium">#5 {{tracks.4.title}}</div>
  311. {% if tracks.4 %}
  312. {% if tracks.4.album.cover_image %}
  313. <a href="{{tracks.4.get_absolute_url}}"><img src="{{tracks.4.album.cover_image.url}}" width="150px"></a>
  314. {% else %}
  315. <a href="{{tracks.4.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="150px"></a>
  316. {% endif %}
  317. {% endif %}
  318. </div>
  319. </div>
  320. </div>
  321. <div style="float:left; width:300px;">
  322. <div style="display:flex; flex-wrap: wrap;">
  323. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  324. <div class="caption-small">#6 {{tracks.5.title}}</div>
  325. {% if tracks.5 %}
  326. {% if tracks.5.album.cover_image %}
  327. <a href="{{tracks.5.get_absolute_url}}"><img src="{{tracks.5.album.cover_image.url}}" width="100px"></a>
  328. {% else %}
  329. <a href="{{tracks.5.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="100px"></a>
  330. {% endif %}
  331. {% endif %}
  332. </div>
  333. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  334. <div class="caption-small">#7 {{tracks.6.title}}</div>
  335. {% if tracks.6 %}
  336. {% if tracks.6.album.cover_image %}
  337. <a href="{{tracks.6.get_absolute_url}}"><img src="{{tracks.6.album.cover_image.url}}" width="100px"></a>
  338. {% else %}
  339. <a href="{{tracks.6.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="100px"></a>
  340. {% endif %}
  341. {% endif %}
  342. </div>
  343. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  344. <div class="caption-small">#8 {{tracks.7.title}}</div>
  345. {% if tracks.7 %}
  346. {% if tracks.7.album.cover_image %}
  347. <a href="{{tracks.7.get_absolute_url}}"><img src="{{tracks.7.album.cover_image.url}}" width="100px"></a>
  348. {% else %}
  349. <a href="{{tracks.7.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="100px"></a>
  350. {% endif %}
  351. {% endif %}
  352. </div>
  353. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  354. <div class="caption-small">#9 {{tracks.8.title}}</div>
  355. {% if tracks.8 %}
  356. {% if tracks.8.album.cover_image %}
  357. <a href="{{tracks.8.get_absolute_url}}"><img src="{{tracks.8.album.cover_image.url}}" width="100px"></a>
  358. {% else %}
  359. <a href="{{tracks.8.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="100px"></a>
  360. {% endif %}
  361. {% endif %}
  362. </div>
  363. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  364. <div class="caption-small">#10 {{tracks.9.title}}</div>
  365. {% if tracks.9 %}
  366. {% if tracks.9.album.cover_image %}
  367. <a href="{{tracks.9.get_absolute_url}}"><img src="{{tracks.9.album.cover_image.url}}" width="100px"></a>
  368. {% else %}
  369. <a href="{{tracks.9.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="100px"></a>
  370. {% endif %}
  371. {% endif %}
  372. </div>
  373. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  374. <div class="caption-small">#11 {{tracks.10.title}}</div>
  375. {% if tracks.10 %}
  376. {% if tracks.10.album.cover_image %}
  377. <a href="{{tracks.10.get_absolute_url}}"><img src="{{tracks.10.album.cover_image.url}}" width="100px"></a>
  378. {% else %}
  379. <a href="{{tracks.10.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="100px"></a>
  380. {% endif %}
  381. {% endif %}
  382. </div>
  383. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  384. <div class="caption-small">#12 {{tracks.11.title}}</div>
  385. {% if tracks.11 %}
  386. {% if tracks.11.album.cover_image %}
  387. <a href="{{tracks.11.get_absolute_url}}"><img src="{{tracks.11.album.cover_image.url}}" width="100px"></a>
  388. {% else %}
  389. <a href="{{tracks.11.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="100px"></a>
  390. {% endif %}
  391. {% endif %}
  392. </div>
  393. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  394. <div class="caption-small">#13 {{tracks.12.title}}</div>
  395. {% if tracks.12 %}
  396. {% if tracks.12.album.cover_image %}
  397. <a href="{{tracks.12.get_absolute_url}}"><img src="{{tracks.12.album.cover_image.url}}" width="100px"></a>
  398. {% else %}
  399. <a href="{{tracks.12.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="100px"></a>
  400. {% endif %}
  401. {% endif %}
  402. </div>
  403. <div class="image-wrapper" class="image-wrapper" style="width:33;">
  404. <div class="caption-small">#14 {{tracks.13.title}}</div>
  405. {% if tracks.13 %}
  406. {% if tracks.13.album.cover_image %}
  407. <a href="{{tracks.13.get_absolute_url}}"><img src="{{tracks.13.album.cover_image.url}}" width="100px"></a>
  408. {% else %}
  409. <a href="{{tracks.13.get_absolute_url}}"><img src="{% static 'images/not-found.jpg' %}" width="100px"></a>
  410. {% endif %}
  411. {% endif %}
  412. </div>
  413. </div>
  414. </div>
  415. </div>
  416. </div>
  417. {% endfor %}
  418. </div>
  419. </div>
  420. <div class="row">
  421. <h2>Last Scrobbles</h2>
  422. <p>Today <b>{{counts.today}}</b> | This Week <b>{{counts.week}}</b> | This Month <b>{{counts.month}}</b> | This Year <b>{{counts.year}}</b> | All Time <b>{{counts.alltime}}</b></p>
  423. </div>
  424. <div class="row">
  425. <ul class="nav nav-tabs" id="myTab" role="tablist">
  426. <li class="nav-item" role="presentation">
  427. <button class="nav-link active" id="home-tab" data-bs-toggle="tab" data-bs-target="#latest-listened"
  428. type="button" role="tab" aria-controls="home" aria-selected="true">Tracks</button>
  429. </li>
  430. <li class="nav-item" role="presentation">
  431. <button class="nav-link" id="profile-tab" data-bs-toggle="tab" data-bs-target="#latest-watched"
  432. type="button" role="tab" aria-controls="profile" aria-selected="false">Videos</button>
  433. </li>
  434. <li class="nav-item" role="presentation">
  435. <button class="nav-link" id="profile-tab" data-bs-toggle="tab" data-bs-target="#latest-podcasted"
  436. type="button" role="tab" aria-controls="profile" aria-selected="false">Podcasts</button>
  437. </li>
  438. <li class="nav-item" role="presentation">
  439. <button class="nav-link" id="profile-tab" data-bs-toggle="tab" data-bs-target="#latest-sports"
  440. type="button" role="tab" aria-controls="profile" aria-selected="false">Sports</button>
  441. </li>
  442. <li class="nav-item" role="presentation">
  443. <button class="nav-link" id="profile-tab" data-bs-toggle="tab" data-bs-target="#latest-videogames"
  444. type="button" role="tab" aria-controls="profile" aria-selected="false">Video Games</button>
  445. </li>
  446. <li class="nav-item" role="presentation">
  447. <button class="nav-link" id="profile-tab" data-bs-toggle="tab" data-bs-target="#latest-boardgames"
  448. type="button" role="tab" aria-controls="profile" aria-selected="false">Board Games</button>
  449. </li>
  450. </ul>
  451. <div class="tab-content" id="myTabContent2">
  452. <div class="tab-pane fade show active" id="latest-listened" role="tabpanel"
  453. aria-labelledby="latest-listened-tab">
  454. <div class="table-responsive">
  455. <table class="table table-striped table-sm">
  456. <thead>
  457. <tr>
  458. <th scope="col">Time</th>
  459. <th scope="col">Album</th>
  460. <th scope="col">Track</th>
  461. <th scope="col">Artist</th>
  462. </tr>
  463. </thead>
  464. <tbody>
  465. {% for scrobble in object_list %}
  466. <tr>
  467. <td>{{scrobble.timestamp|naturaltime}}</td>
  468. {% if scrobble.track.album.cover_image %}
  469. <td><a href="{{scrobble.track.album.get_absolute_url}}"><img src="{{scrobble.track.album.cover_image.url}}" width=25 height=25 style="border:1px solid black;" /></aa></td>
  470. {% else %}
  471. <td><a href="{{scrobble.track.album.get_absolute_url}}">{{scrobble.track.album.name}}</a></td>
  472. {% endif %}
  473. <td><a href="{{scrobble.track.get_absolute_url }}">{{scrobble.track.title}}</a></td>
  474. <td><a href="{{scrobble.track.artist.get_absolute_url }}">{{scrobble.track.artist.name}}</aa></td>
  475. </tr>
  476. {% endfor %}
  477. </tbody>
  478. </table>
  479. </div>
  480. </div>
  481. <div class="tab-pane fade show" id="latest-watched" role="tabpanel"
  482. aria-labelledby="latest-watched-tab">
  483. <h2>Latest watched</h2>
  484. <div class="table-responsive">
  485. <table class="table table-striped table-sm">
  486. <thead>
  487. <tr>
  488. <th scope="col">Time</th>
  489. <th scope="col">Title</th>
  490. <th scope="col">Series</th>
  491. </tr>
  492. </thead>
  493. <tbody>
  494. {% for scrobble in video_scrobble_list %}
  495. <tr>
  496. <td>{{scrobble.timestamp|naturaltime}}</td>
  497. <td><a href="{{scrobble.video.get_absolute_url }}">{% if scrobble.video.tv_series%}S{{scrobble.video.season_number}}E{{scrobble.video.episode_number}} -{%endif %} {{scrobble.video.title}}</a></td>
  498. <td><a href="{{scrobble.video.tv_series.get_absolute_url }}">{% if scrobble.video.tv_series %}{{scrobble.video.tv_series}}</a>{% endif %}
  499. </td>
  500. </tr>
  501. {% endfor %}
  502. </tbody>
  503. </table>
  504. </div>
  505. </div>
  506. <div class="tab-pane fade show" id="latest-sports" role="tabpanel" aria-labelledby="latest-sports-tab">
  507. <h2>Latest Sports</h2>
  508. <div class="table-responsive">
  509. <table class="table table-striped table-sm">
  510. <thead>
  511. <tr>
  512. <th scope="col">Date</th>
  513. <th scope="col">Title</th>
  514. <th scope="col">Round</th>
  515. <th scope="col">League</th>
  516. </tr>
  517. </thead>
  518. <tbody>
  519. {% for scrobble in sport_scrobble_list %}
  520. <tr>
  521. <td>{{scrobble.timestamp|naturaltime}}</td>
  522. <td>{{scrobble.sport_event.title}}</td>
  523. <td>{{scrobble.sport_event.round.name}}</td>
  524. <td>{{scrobble.sport_event.round.season.league}}</td>
  525. </tr>
  526. {% endfor %}
  527. </tbody>
  528. </table>
  529. </div>
  530. </div>
  531. <div class="tab-pane fade show" id="latest-podcasted" role="tabpanel"
  532. aria-labelledby="latest-podcasted-tab">
  533. <h2>Latest Podcasted</h2>
  534. <div class="table-responsive">
  535. <table class="table table-striped table-sm">
  536. <thead>
  537. <tr>
  538. <th scope="col">Date</th>
  539. <th scope="col">Title</th>
  540. <th scope="col">Podcast</th>
  541. </tr>
  542. </thead>
  543. <tbody>
  544. {% for scrobble in podcast_scrobble_list %}
  545. <tr>
  546. <td>{{scrobble.timestamp|naturaltime}}</td>
  547. <td>{{scrobble.podcast_episode.title}}</td>
  548. <td>{{scrobble.podcast_episode.podcast}}</td>
  549. </tr>
  550. {% endfor %}
  551. </tbody>
  552. </table>
  553. </div>
  554. </div>
  555. <div class="tab-pane fade show" id="latest-videogames" role="tabpanel"
  556. aria-labelledby="latest-videogames-tab">
  557. <h2>Latest Video Games</h2>
  558. <div class="table-responsive">
  559. <table class="table table-striped table-sm">
  560. <thead>
  561. <tr>
  562. <th scope="col">Date</th>
  563. <th scope="col">Cover</th>
  564. <th scope="col">Title</th>
  565. <th scope="col">Score</th>
  566. </tr>
  567. </thead>
  568. <tbody>
  569. {% for scrobble in videogame_scrobble_list %}
  570. <tr>
  571. <td>{{scrobble.timestamp|naturaltime}}</td>
  572. {% if scrobble.videogame_screenshot %}
  573. <td><a href="{{scrobble.media_obj.get_absolute_url}}"><img src="{{scrobble.videogame_screenshot.url}}" width=25 height=25 style="border:1px solid black;" /></aa></td>
  574. {% else %}
  575. <td><a href="{{scrobble.media_obj.get_absolute_url}}"><img src="{{scrobble.media_obj.primary_image_url}}" width=25 height=25 style="border:1px solid black;" /></aa></td>
  576. {% endif %}
  577. <td>{{scrobble.media_obj.title}}</td>
  578. <td>{{scrobble.media_obj.hltb_score}}</td>
  579. </tr>
  580. {% endfor %}
  581. </tbody>
  582. </table>
  583. </div>
  584. </div>
  585. <div class="tab-pane fade show" id="latest-boardgames" role="tabpanel"
  586. aria-labelledby="latest-boardgames-tab">
  587. <h2>Latest Board Games</h2>
  588. <div class="table-responsive">
  589. <table class="table table-striped table-sm">
  590. <thead>
  591. <tr>
  592. <th scope="col">Date</th>
  593. <th scope="col">Cover</th>
  594. <th scope="col">Title</th>
  595. <th scope="col">Rating</th>
  596. </tr>
  597. </thead>
  598. <tbody>
  599. {% for scrobble in boardgame_scrobble_list %}
  600. <tr>
  601. <td>{{scrobble.timestamp|naturaltime}}</td>
  602. <td><a href="{{scrobble.media_obj.get_absolute_url}}"><img src="{{scrobble.media_obj.primary_image_url}}" width=25 height=25 style="border:1px solid black;" /></aa></td>
  603. <td>{{scrobble.media_obj.title}}</td>
  604. <td>{{scrobble.media_obj.rating}}</td>
  605. </tr>
  606. {% endfor %}
  607. </tbody>
  608. </table>
  609. </div>
  610. </div>
  611. </div>
  612. {% endif %}
  613. </div>
  614. </main>
  615. <div class="modal fade" id="importModal" tabindex="-1" role="dialog" aria-labelledby="importModalLabel"
  616. aria-hidden="true">
  617. <div class="modal-dialog" role="document">
  618. <div class="modal-content">
  619. <div class="modal-header">
  620. <h5 class="modal-title" id="importModalLabel">Import scrobbles</h5>
  621. <button type="button" class="close" data-bs-dismiss="modal" aria-label="Close">
  622. <div aria-hidden="true">&times;</div>
  623. </button>
  624. </div>
  625. <form action="{% url 'scrobbles:audioscrobbler-file-upload' %}" method="post" enctype="multipart/form-data">
  626. <div class="modal-body">
  627. {% csrf_token %}
  628. <div class="form-group">
  629. <label for="tsv_file" class="col-form-label">Audioscrobbler TSV file:</label>
  630. <input type="file" name="tsv_file" class="form-control" id="id_tsv_file">
  631. </div>
  632. </div>
  633. <div class="modal-footer">
  634. <button type="submit" class="btn btn-primary">Import</button>
  635. </div>
  636. </form>
  637. <form action="{% url 'scrobbles:koreader-file-upload' %}" method="post" enctype="multipart/form-data">
  638. <div class="modal-body">
  639. {% csrf_token %}
  640. <div class="form-group">
  641. <label for="tsv_file" class="col-form-label">KOReader sqlite file:</label>
  642. <input type="file" name="sqlite_file" class="form-control" id="id_sqlite_file">
  643. </div>
  644. </div>
  645. <div class="modal-footer">
  646. <button type="submit" class="btn btn-primary">Import</button>
  647. </div>
  648. </form>
  649. </div>
  650. </div>
  651. </div>
  652. <div class="modal fade" id="exportModal" tabindex="-1" role="dialog" aria-labelledby="exportModalLabel"
  653. aria-hidden="true">
  654. <div class="modal-dialog" role="document">
  655. <div class="modal-content">
  656. <div class="modal-header">
  657. <h5 class="modal-title" id="exportModalLabel">Export scrobbles</h5>
  658. <button type="button" class="close" data-bs-dismiss="modal" aria-label="Close">
  659. <div aria-hidden="true">&times;</div>
  660. </button>
  661. </div>
  662. <form action="{% url 'scrobbles:export' %}" method="get">
  663. <div class="modal-body">
  664. {% csrf_token %}
  665. <div class="form-group">
  666. {{export_form.as_div}}
  667. </div>
  668. </div>
  669. <div class="modal-footer">
  670. <button type="submit" class="btn btn-primary">Export</button>
  671. <button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Close</button>
  672. </div>
  673. </form>
  674. </div>
  675. </div>
  676. </div>
  677. {% endblock %}
  678. {% block extra_js %}
  679. <script src="https://cdn.jsdelivr.net/npm/feather-icons@4.28.0/dist/feather.min.js" integrity="sha384-uO3SXW5IuS1ZpFPKugNNWqTZRRglnUJK6UAZ/gxOX80nxEkN9NcGZTftn6RzhGWE" crossorigin="anonymous"></script><script src="https://cdn.jsdelivr.net/npm/chart.js@2.9.4/dist/Chart.min.js" integrity="sha384-zNy6FEbO50N+Cg5wap8IKA4M/ZnLJgzc6w2NqACZaK0u0FXfOWRRJOnQtpZun8ha" crossorigin="anonymous"></script>
  680. <script>
  681. /* globals Chart:false, feather:false */
  682. (function () {
  683. 'use strict'
  684. feather.replace({ 'aria-hidden': 'true' })
  685. // Graphs
  686. var ctx = document.getElementById('myChart')
  687. // eslint-disable-next-line no-unused-vars
  688. var myChart = new Chart(ctx, {
  689. type: 'line',
  690. data: {
  691. labels: [
  692. {% for day in weekly_data.keys %}
  693. "{{day}}"{% if not forloop.last %},{% endif %}
  694. {% endfor %}
  695. ],
  696. datasets: [{
  697. data: [
  698. {% for count in weekly_data.values %}
  699. {{count}}{% if not forloop.last %},{% endif %}
  700. {% endfor %}
  701. ],
  702. lineTension: 0,
  703. backgroundColor: 'transparent',
  704. borderColor: '#007bf0',
  705. borderWidth: 4,
  706. pointBackgroundColor: '#007bff'
  707. }]
  708. },
  709. options: {
  710. scales: {
  711. yAxes: [{
  712. ticks: {
  713. beginAtZero: true
  714. }
  715. }]
  716. },
  717. legend: {
  718. display: false
  719. }
  720. }
  721. })
  722. })()
  723. </script>
  724. {% endblock %}