@saooti/octopus-sdk 1.0.0 → 1.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (317) hide show
  1. package/README.md +261 -0
  2. package/index.d.ts +4 -0
  3. package/index.ts +55 -3
  4. package/package.json +53 -45
  5. package/public/config.ts +4 -0
  6. package/public/css/fonts/icomoon.eot +0 -0
  7. package/public/css/fonts/icomoon.svg +129 -0
  8. package/public/css/fonts/icomoon.ttf +0 -0
  9. package/public/css/fonts/icomoon.woff +0 -0
  10. package/public/css/fonts/style.css +186 -1224
  11. package/public/img/403.jpeg +0 -0
  12. package/public/img/404.svg +242 -0
  13. package/public/img/ACPM.png +0 -0
  14. package/public/img/ouest_france_logo.svg +12 -0
  15. package/src/App.vue +102 -31
  16. package/src/api/classicCrud.ts +34 -0
  17. package/src/api/initialize.ts +5 -2
  18. package/src/assets/bootstrap-diff.scss +197 -0
  19. package/src/assets/form.scss +77 -77
  20. package/src/assets/general.scss +284 -272
  21. package/src/assets/live.scss +39 -0
  22. package/src/assets/modal.scss +43 -50
  23. package/src/assets/multiselect.scss +522 -573
  24. package/src/assets/octopus-library.scss +18 -4
  25. package/src/assets/share.scss +243 -0
  26. package/src/assets/transition.scss +98 -2
  27. package/src/components/display/aggregator/RssSection.vue +3 -5
  28. package/src/components/display/categories/CategoryChooser.vue +151 -114
  29. package/src/components/display/categories/CategoryFilter.vue +162 -0
  30. package/src/components/display/categories/CategoryList.vue +149 -117
  31. package/src/components/display/comments/AddCommentModal.vue +103 -108
  32. package/src/components/display/comments/CommentBasicView.vue +95 -0
  33. package/src/components/display/comments/CommentInput.vue +195 -183
  34. package/src/components/display/comments/CommentItem.vue +170 -208
  35. package/src/components/display/comments/CommentList.vue +134 -228
  36. package/src/components/display/comments/CommentParentInfo.vue +28 -62
  37. package/src/components/display/comments/CommentPlayer.vue +53 -56
  38. package/src/components/display/comments/CommentSection.vue +52 -83
  39. package/src/components/display/edit/EditBox.vue +10 -4
  40. package/src/components/display/edit/EditCommentBox.vue +124 -38
  41. package/src/components/display/emission/EmissionChooser.vue +88 -108
  42. package/src/components/display/emission/EmissionInlineList.vue +156 -157
  43. package/src/components/display/emission/EmissionItem.vue +72 -226
  44. package/src/components/display/emission/EmissionList.vue +159 -201
  45. package/src/components/display/emission/EmissionPlayerItem.vue +203 -174
  46. package/src/components/display/filter/AdvancedSearch.vue +224 -432
  47. package/src/components/display/filter/CategoryFilter.vue +105 -0
  48. package/src/components/display/filter/MonetizableFilter.vue +18 -15
  49. package/src/components/display/filter/ProductorSearch.vue +153 -176
  50. package/src/components/display/filter/RubriqueChoice.vue +97 -0
  51. package/src/components/display/filter/RubriqueFilter.vue +232 -0
  52. package/src/components/display/list/ListPaginate.vue +145 -0
  53. package/src/components/display/list/Paginate.vue +218 -0
  54. package/src/components/display/live/CountDown.vue +31 -30
  55. package/src/components/display/live/LiveHorizontalList.vue +84 -62
  56. package/src/components/display/live/LiveItem.vue +161 -186
  57. package/src/components/display/live/LiveList.vue +184 -295
  58. package/src/components/display/organisation/OrganisationChooser.vue +124 -162
  59. package/src/components/display/organisation/OrganisationChooserLight.vue +53 -56
  60. package/src/components/display/participant/ParticipantItem.vue +91 -128
  61. package/src/components/display/participant/ParticipantList.vue +134 -121
  62. package/src/components/display/playlist/PlaylistItem.vue +60 -97
  63. package/src/components/display/playlist/PlaylistList.vue +99 -92
  64. package/src/components/display/playlist/PodcastList.vue +117 -132
  65. package/src/components/display/playlist/PodcastPlaylistInlineList.vue +191 -0
  66. package/src/components/display/podcasts/AnimatorsItem.vue +29 -37
  67. package/src/components/display/podcasts/ParticipantDescription.vue +82 -0
  68. package/src/components/display/podcasts/PodcastFilterList.vue +49 -102
  69. package/src/components/display/podcasts/PodcastImage.vue +208 -225
  70. package/src/components/display/podcasts/PodcastInlineList.vue +62 -355
  71. package/src/components/display/podcasts/PodcastInlineListClassic.vue +240 -0
  72. package/src/components/display/podcasts/PodcastInlineListTemplate.vue +167 -0
  73. package/src/components/display/podcasts/PodcastItem.vue +94 -218
  74. package/src/components/display/podcasts/PodcastItemInfo.vue +166 -0
  75. package/src/components/display/podcasts/PodcastList.vue +127 -178
  76. package/src/components/display/podcasts/PodcastModuleBox.vue +234 -0
  77. package/src/components/display/podcasts/PodcastPlayBar.vue +101 -0
  78. package/src/components/display/podcasts/PodcastSwiperList.vue +204 -0
  79. package/src/components/display/podcasts/TagList.vue +60 -38
  80. package/src/components/display/rubriques/RubriqueChooser.vue +144 -131
  81. package/src/components/display/rubriques/RubriqueList.vue +216 -0
  82. package/src/components/display/sharing/PlayerParameters.vue +111 -153
  83. package/src/components/display/sharing/QrCode.vue +96 -0
  84. package/src/components/display/sharing/ShareButtons.vue +63 -233
  85. package/src/components/display/sharing/ShareButtonsIntern.vue +225 -0
  86. package/src/components/display/sharing/ShareDistribution.vue +102 -166
  87. package/src/components/display/sharing/SharePlayer.vue +189 -353
  88. package/src/components/display/sharing/SharePlayerColors.vue +78 -0
  89. package/src/components/display/sharing/SharePlayerTypes.vue +112 -0
  90. package/src/components/display/sharing/SplitButton.vue +42 -0
  91. package/src/components/display/sharing/SubscribeButtons.vue +65 -118
  92. package/src/components/display/studio/RecordingItemButton.vue +3 -5
  93. package/src/components/form/ClassicCheckbox.vue +66 -0
  94. package/src/components/form/ClassicLoading.vue +28 -0
  95. package/src/components/form/ClassicRadio.vue +66 -0
  96. package/src/components/form/ClassicSearch.vue +86 -0
  97. package/src/components/form/ClassicSelect.vue +65 -0
  98. package/src/components/misc/ErrorMessage.vue +27 -18
  99. package/src/components/misc/Footer.vue +151 -108
  100. package/src/components/misc/HomeDropdown.vue +95 -154
  101. package/src/components/misc/LeftMenu.vue +137 -206
  102. package/src/components/misc/Popover.vue +118 -0
  103. package/src/components/misc/Snackbar.vue +73 -37
  104. package/src/components/misc/TopBar.vue +211 -314
  105. package/src/components/misc/modal/ClipboardModal.vue +34 -47
  106. package/src/components/misc/modal/MessageModal.vue +74 -53
  107. package/src/components/misc/modal/NewsletterModal.vue +272 -224
  108. package/src/components/misc/modal/QrCodeModal.vue +58 -0
  109. package/src/components/misc/modal/ShareModalPlayer.vue +140 -74
  110. package/src/components/misc/player/Player.vue +146 -0
  111. package/src/components/misc/player/PlayerCompact.vue +154 -0
  112. package/src/components/misc/player/PlayerLarge.vue +166 -0
  113. package/src/components/misc/player/PlayerProgressBar.vue +112 -0
  114. package/src/components/misc/player/PlayerTimeline.vue +61 -0
  115. package/src/components/mixins/functions.ts +21 -18
  116. package/src/components/mixins/handle403.ts +17 -0
  117. package/src/components/mixins/init.ts +24 -0
  118. package/src/components/mixins/orgaComputed.ts +15 -0
  119. package/src/components/mixins/organisationFilter.ts +41 -0
  120. package/src/components/mixins/player/playerComment.ts +98 -0
  121. package/src/components/mixins/player/playerDisplay.ts +113 -0
  122. package/src/components/mixins/player/playerLive.ts +111 -0
  123. package/src/components/mixins/player/playerLogic.ts +225 -0
  124. package/src/components/mixins/tagOfMixins.ts +19 -0
  125. package/src/components/pages/Category.vue +24 -28
  126. package/src/components/pages/Emission.vue +144 -177
  127. package/src/components/pages/Emissions.vue +94 -110
  128. package/src/components/pages/Error403Page.vue +60 -0
  129. package/src/components/pages/Home.vue +74 -21
  130. package/src/components/pages/Lives.vue +53 -58
  131. package/src/components/pages/PageNotFound.vue +55 -0
  132. package/src/components/pages/Participant.vue +113 -129
  133. package/src/components/pages/Participants.vue +25 -41
  134. package/src/components/pages/Playlist.vue +86 -86
  135. package/src/components/pages/Playlists.vue +33 -52
  136. package/src/components/pages/Podcast.vue +206 -478
  137. package/src/components/pages/Podcasts.vue +112 -156
  138. package/src/components/pages/Rubrique.vue +21 -27
  139. package/src/components/pages/Search.vue +37 -94
  140. package/src/helper/dom.ts +2 -2
  141. package/src/helper/duration.ts +18 -8
  142. package/src/helper/useEventListener.ts +18 -0
  143. package/src/i18n.ts +36 -0
  144. package/src/locale/de.ts +309 -0
  145. package/src/locale/educationen.ts +14 -0
  146. package/src/locale/en.ts +316 -1
  147. package/src/locale/es.ts +309 -0
  148. package/src/locale/fr.ts +47 -15
  149. package/src/locale/it.ts +307 -0
  150. package/src/locale/messages.ts +11 -2
  151. package/src/locale/sl.ts +309 -0
  152. package/src/main.ts +64 -55
  153. package/src/router/router.ts +192 -159
  154. package/src/sass/_variables.scss +10 -8
  155. package/src/shims-tsx.d.ts +15 -0
  156. package/src/shims-vue-recaptcha-v3.d.ts +9 -0
  157. package/src/shims-vue.d.ts +5 -6
  158. package/src/store/AppStore.ts +41 -173
  159. package/src/store/class/adserver/adserverConfig.ts +9 -0
  160. package/src/store/class/adserver/adserverOtherEmission.ts +13 -0
  161. package/src/store/class/adserver/adserverTiming.ts +10 -0
  162. package/src/store/class/cartouchier/cartouche.ts +15 -0
  163. package/src/store/class/cartouchier/cartouchier.ts +9 -0
  164. package/src/store/class/conference/conference.ts +35 -0
  165. package/src/store/class/conference/conferenceMessage.ts +10 -0
  166. package/src/store/class/conference/conferenceParticipant.ts +18 -0
  167. package/src/store/class/conference/pad.ts +15 -0
  168. package/src/store/class/conference/studioCall.ts +7 -0
  169. package/src/store/class/contract/contract.ts +7 -0
  170. package/src/store/class/contract/contractOrganisation.ts +7 -0
  171. package/src/store/class/ftp/ftpEmission.ts +41 -0
  172. package/src/store/class/ftp/ftpParam.ts +12 -0
  173. package/src/store/class/ftp/testFtpEmission.ts +5 -0
  174. package/src/store/class/general/audioView.ts +21 -0
  175. package/src/store/class/general/category.ts +10 -0
  176. package/src/store/class/general/comment.ts +16 -0
  177. package/src/store/class/general/customPlayer.ts +9 -0
  178. package/src/store/class/general/emission.ts +23 -0
  179. package/src/store/class/general/fetchParam.ts +3 -0
  180. package/src/store/class/general/initState.ts +25 -0
  181. package/src/store/class/general/interfacePageable.ts +16 -0
  182. package/src/store/class/general/ituneCategory.ts +5 -0
  183. package/src/store/class/general/media.ts +14 -0
  184. package/src/store/class/general/organisation.ts +20 -0
  185. package/src/store/class/general/pageable.ts +13 -0
  186. package/src/store/class/general/participant.ts +12 -0
  187. package/src/store/class/general/player.ts +14 -0
  188. package/src/store/class/general/playlist.ts +13 -0
  189. package/src/store/class/general/podcast.ts +43 -0
  190. package/src/store/class/general/sortPageable.ts +5 -0
  191. package/src/store/class/general/soundcastCategory.ts +8 -0
  192. package/src/store/class/ouestFrance/ofTag.ts +36 -0
  193. package/src/store/class/ouestFrance/ofTagInfo.ts +9 -0
  194. package/src/store/class/ouestFrance/ofTagPage.ts +7 -0
  195. package/src/store/class/ouestFrance/ofTagSeo.ts +7 -0
  196. package/src/store/class/ouestFrance/ofTagVente.ts +6 -0
  197. package/src/store/class/ouestFrance/ofTagWithParents.ts +26 -0
  198. package/src/store/class/rss/aggregator.ts +28 -0
  199. package/src/store/class/rss/rssEmission.ts +14 -0
  200. package/src/store/class/rss/rssInfo.ts +8 -0
  201. package/src/store/class/rubrique/rubriquage.ts +10 -0
  202. package/src/store/class/rubrique/rubriquageFilter.ts +6 -0
  203. package/src/store/class/rubrique/rubrique.ts +9 -0
  204. package/src/store/class/stat/statArrayIncome.ts +6 -0
  205. package/src/store/class/stat/statArrayObject.ts +34 -0
  206. package/src/store/class/stat/statGraph.ts +7 -0
  207. package/src/store/class/user/person.ts +15 -0
  208. package/src/store/class/user/profile.ts +12 -0
  209. package/src/store/class/user/userKeycloak.ts +24 -0
  210. package/src/store/paramStore.ts +222 -217
  211. package/src/store/typeAppStore.ts +186 -235
  212. package/src/vuex-shim.d.ts +8 -0
  213. package/tsconfig.json +4 -0
  214. package/vue.config.js +16 -0
  215. package/public/css/fonts/saooti-ui.eot +0 -0
  216. package/public/css/fonts/saooti-ui.svg +0 -829
  217. package/public/css/fonts/saooti-ui.ttf +0 -0
  218. package/public/css/fonts/saooti-ui.woff +0 -0
  219. package/public/css/fonts/stitcher-logo.svg +0 -83
  220. package/public/img/ACPM.PNG +0 -0
  221. package/src/api/comments.ts +0 -7
  222. package/src/api/emissions.ts +0 -7
  223. package/src/api/podcasts.ts +0 -7
  224. package/src/api/profile.ts +0 -8
  225. package/src/api/studio.ts +0 -1
  226. package/src/assets/bootstrap_scss/_alert.scss +0 -51
  227. package/src/assets/bootstrap_scss/_badge.scss +0 -54
  228. package/src/assets/bootstrap_scss/_breadcrumb.scss +0 -42
  229. package/src/assets/bootstrap_scss/_button-group.scss +0 -164
  230. package/src/assets/bootstrap_scss/_buttons.scss +0 -291
  231. package/src/assets/bootstrap_scss/_card.scss +0 -278
  232. package/src/assets/bootstrap_scss/_carousel.scss +0 -197
  233. package/src/assets/bootstrap_scss/_close.scss +0 -41
  234. package/src/assets/bootstrap_scss/_code.scss +0 -48
  235. package/src/assets/bootstrap_scss/_custom-forms.scss +0 -522
  236. package/src/assets/bootstrap_scss/_dropdown.scss +0 -201
  237. package/src/assets/bootstrap_scss/_forms.scss +0 -352
  238. package/src/assets/bootstrap_scss/_functions.scss +0 -134
  239. package/src/assets/bootstrap_scss/_grid.scss +0 -69
  240. package/src/assets/bootstrap_scss/_images.scss +0 -42
  241. package/src/assets/bootstrap_scss/_input-group.scss +0 -191
  242. package/src/assets/bootstrap_scss/_jumbotron.scss +0 -17
  243. package/src/assets/bootstrap_scss/_list-group.scss +0 -158
  244. package/src/assets/bootstrap_scss/_media.scss +0 -8
  245. package/src/assets/bootstrap_scss/_mixins.scss +0 -47
  246. package/src/assets/bootstrap_scss/_modal.scss +0 -243
  247. package/src/assets/bootstrap_scss/_nav.scss +0 -120
  248. package/src/assets/bootstrap_scss/_navbar.scss +0 -324
  249. package/src/assets/bootstrap_scss/_pagination.scss +0 -74
  250. package/src/assets/bootstrap_scss/_popover.scss +0 -170
  251. package/src/assets/bootstrap_scss/_print.scss +0 -141
  252. package/src/assets/bootstrap_scss/_progress.scss +0 -46
  253. package/src/assets/bootstrap_scss/_reboot.scss +0 -482
  254. package/src/assets/bootstrap_scss/_root.scss +0 -20
  255. package/src/assets/bootstrap_scss/_spinners.scss +0 -55
  256. package/src/assets/bootstrap_scss/_tables.scss +0 -185
  257. package/src/assets/bootstrap_scss/_toasts.scss +0 -44
  258. package/src/assets/bootstrap_scss/_tooltip.scss +0 -115
  259. package/src/assets/bootstrap_scss/_transitions.scss +0 -20
  260. package/src/assets/bootstrap_scss/_type.scss +0 -125
  261. package/src/assets/bootstrap_scss/_utilities.scss +0 -17
  262. package/src/assets/bootstrap_scss/_variables.scss +0 -1145
  263. package/src/assets/bootstrap_scss/bootstrap-grid.scss +0 -29
  264. package/src/assets/bootstrap_scss/bootstrap-reboot.scss +0 -12
  265. package/src/assets/bootstrap_scss/bootstrap.scss +0 -44
  266. package/src/assets/bootstrap_scss/mixins/_alert.scss +0 -13
  267. package/src/assets/bootstrap_scss/mixins/_background-variant.scss +0 -22
  268. package/src/assets/bootstrap_scss/mixins/_badge.scss +0 -17
  269. package/src/assets/bootstrap_scss/mixins/_border-radius.scss +0 -63
  270. package/src/assets/bootstrap_scss/mixins/_box-shadow.scss +0 -20
  271. package/src/assets/bootstrap_scss/mixins/_breakpoints.scss +0 -123
  272. package/src/assets/bootstrap_scss/mixins/_buttons.scss +0 -110
  273. package/src/assets/bootstrap_scss/mixins/_caret.scss +0 -62
  274. package/src/assets/bootstrap_scss/mixins/_clearfix.scss +0 -7
  275. package/src/assets/bootstrap_scss/mixins/_deprecate.scss +0 -10
  276. package/src/assets/bootstrap_scss/mixins/_float.scss +0 -14
  277. package/src/assets/bootstrap_scss/mixins/_forms.scss +0 -177
  278. package/src/assets/bootstrap_scss/mixins/_gradients.scss +0 -45
  279. package/src/assets/bootstrap_scss/mixins/_grid-framework.scss +0 -71
  280. package/src/assets/bootstrap_scss/mixins/_grid.scss +0 -69
  281. package/src/assets/bootstrap_scss/mixins/_hover.scss +0 -37
  282. package/src/assets/bootstrap_scss/mixins/_image.scss +0 -36
  283. package/src/assets/bootstrap_scss/mixins/_list-group.scss +0 -21
  284. package/src/assets/bootstrap_scss/mixins/_lists.scss +0 -7
  285. package/src/assets/bootstrap_scss/mixins/_nav-divider.scss +0 -11
  286. package/src/assets/bootstrap_scss/mixins/_pagination.scss +0 -22
  287. package/src/assets/bootstrap_scss/mixins/_reset-text.scss +0 -17
  288. package/src/assets/bootstrap_scss/mixins/_resize.scss +0 -6
  289. package/src/assets/bootstrap_scss/mixins/_screen-reader.scss +0 -34
  290. package/src/assets/bootstrap_scss/mixins/_size.scss +0 -7
  291. package/src/assets/bootstrap_scss/mixins/_table-row.scss +0 -39
  292. package/src/assets/bootstrap_scss/mixins/_text-emphasis.scss +0 -17
  293. package/src/assets/bootstrap_scss/mixins/_text-hide.scss +0 -11
  294. package/src/assets/bootstrap_scss/mixins/_text-truncate.scss +0 -8
  295. package/src/assets/bootstrap_scss/mixins/_transition.scss +0 -16
  296. package/src/assets/bootstrap_scss/mixins/_visibility.scss +0 -8
  297. package/src/assets/bootstrap_scss/utilities/_align.scss +0 -8
  298. package/src/assets/bootstrap_scss/utilities/_background.scss +0 -19
  299. package/src/assets/bootstrap_scss/utilities/_borders.scss +0 -75
  300. package/src/assets/bootstrap_scss/utilities/_clearfix.scss +0 -3
  301. package/src/assets/bootstrap_scss/utilities/_display.scss +0 -26
  302. package/src/assets/bootstrap_scss/utilities/_embed.scss +0 -39
  303. package/src/assets/bootstrap_scss/utilities/_flex.scss +0 -51
  304. package/src/assets/bootstrap_scss/utilities/_float.scss +0 -11
  305. package/src/assets/bootstrap_scss/utilities/_overflow.scss +0 -5
  306. package/src/assets/bootstrap_scss/utilities/_position.scss +0 -32
  307. package/src/assets/bootstrap_scss/utilities/_screenreaders.scss +0 -11
  308. package/src/assets/bootstrap_scss/utilities/_shadows.scss +0 -6
  309. package/src/assets/bootstrap_scss/utilities/_sizing.scss +0 -20
  310. package/src/assets/bootstrap_scss/utilities/_spacing.scss +0 -73
  311. package/src/assets/bootstrap_scss/utilities/_stretched-link.scss +0 -19
  312. package/src/assets/bootstrap_scss/utilities/_text.scss +0 -72
  313. package/src/assets/bootstrap_scss/utilities/_visibility.scss +0 -13
  314. package/src/assets/bootstrap_scss/vendor/_rfs.scss +0 -204
  315. package/src/components/misc/Player.vue +0 -792
  316. package/src/shims-vuex.d.ts +0 -7
  317. package/src/views/Home.vue +0 -18
@@ -0,0 +1,112 @@
1
+ <template>
2
+ <div
3
+ v-if="!playerError"
4
+ class="progress c-hand"
5
+ @mouseup="seekTo"
6
+ >
7
+ <div
8
+ class="progress-bar bg-light"
9
+ role="progressbar"
10
+ aria-valuenow="0"
11
+ aria-valuemin="0"
12
+ aria-valuemax="100"
13
+ :style="'width: ' + percentLiveProgress + '%'"
14
+ />
15
+ <div
16
+ class="progress-bar primary-bg"
17
+ role="progressbar"
18
+ aria-valuenow="0"
19
+ aria-valuemin="0"
20
+ aria-valuemax="100"
21
+ :style="'width: ' + percentProgress + '%'"
22
+ />
23
+ <div
24
+ v-if="displayAlertBar"
25
+ class="progress-bar progress-bar-duration bg-danger"
26
+ :style="'left: ' + durationLivePosition + '%'"
27
+ />
28
+ </div>
29
+ <CommentPlayer
30
+ v-if="showTimeline"
31
+ :total-time="totalSecondes"
32
+ :comments="comments"
33
+ />
34
+ </template>
35
+
36
+ <script lang="ts">
37
+ import { CommentPodcast } from '@/store/class/general/comment';
38
+ import { defineComponent, defineAsyncComponent } from 'vue';
39
+ const CommentPlayer = defineAsyncComponent(() => import('../../display/comments/CommentPlayer.vue'));
40
+ export default defineComponent({
41
+ name: 'PlayerProgressBar',
42
+
43
+ components: {
44
+ CommentPlayer,
45
+ },
46
+ props: {
47
+ hlsReady: { default: false, type: Boolean},
48
+ showTimeline: { default: false, type: Boolean},
49
+ comments: { default: ()=>[], type: Array as ()=>Array<CommentPodcast>},
50
+ displayAlertBar: { default: false, type: Boolean},
51
+ percentLiveProgress: { default: 0, type: Number},
52
+ durationLivePosition: { default: 0, type: Number},
53
+ playerError: { default: false, type: Boolean},
54
+ listenTime: { default: 0, type: Number},
55
+ },
56
+ emits: ['updateNotListenTime'],
57
+
58
+ computed: {
59
+ percentProgress(): number{
60
+ if(!this.$store.state.player.elapsed){return 0;}
61
+ return this.$store.state.player.elapsed * 100;
62
+ },
63
+ totalSecondes(): number{
64
+ return this.$store.state.player.total;
65
+ },
66
+ },
67
+
68
+
69
+ methods: {
70
+ seekTo(event: MouseEvent): void {
71
+ const audioPlayer: HTMLAudioElement|null = document.querySelector('#audio-player');
72
+ if(!audioPlayer || null===event.currentTarget){return;}
73
+ const rect = (event.currentTarget as Element).getBoundingClientRect();
74
+ const barWidth = (event.currentTarget as Element).clientWidth;
75
+ const x = event.clientX - rect.left;
76
+ const percentPosition = x / barWidth;
77
+ if (percentPosition * 100 >= this.percentLiveProgress) return;
78
+ const seekTime = this.$store.state.player.total * percentPosition;
79
+ this.isSeekTo(audioPlayer, seekTime);
80
+ },
81
+ isSeekTo(audioPlayer: HTMLAudioElement, seekTime: number): void {
82
+ if (this.$store.state.player.podcast || this.$store.state.player.live) {
83
+ this.$emit('updateNotListenTime',seekTime - this.listenTime);
84
+ }
85
+ audioPlayer.currentTime = seekTime;
86
+ }
87
+
88
+ },
89
+ })
90
+ </script>
91
+
92
+ <style lang="scss">
93
+ .octopus-app{
94
+ .progress {
95
+ height: 4px;
96
+ position: relative;
97
+ @media (max-width: 960px) {
98
+ height: 8px;
99
+ }
100
+ }
101
+ .progress-bar-duration {
102
+ width: 10px;
103
+ }
104
+ .progress-bar {
105
+ height: 4px;
106
+ position: absolute;
107
+ @media (max-width: 960px) {
108
+ height: 8px;
109
+ }
110
+ }
111
+ }
112
+ </style>
@@ -0,0 +1,61 @@
1
+ <template>
2
+ <div
3
+ v-if="0 !== comments.length && !isPodcastmaker"
4
+ class="timeline-button"
5
+ @click="$emit('update:showTimeline', !showTimeline)"
6
+ >
7
+ <div
8
+ class="saooti-down"
9
+ :class="showTimeline ? '' : 'arrow-transform'"
10
+ />
11
+ <div>Timeline</div>
12
+ </div>
13
+ </template>
14
+
15
+ <script lang="ts">
16
+ import { state } from '../../../store/paramStore';
17
+ import { defineComponent } from 'vue';
18
+ import { CommentPodcast } from '@/store/class/general/comment';
19
+ export default defineComponent({
20
+ name: 'PlayerTimeline',
21
+
22
+ props: {
23
+ showTimeline: { default: false, type: Boolean},
24
+ comments: { default: ()=>[], type: Array as ()=>Array<CommentPodcast>},
25
+ },
26
+ emits:['update:showTimeline'],
27
+
28
+ computed: {
29
+ isPodcastmaker(): boolean {
30
+ return (state.generalParameters.podcastmaker as boolean);
31
+ },
32
+ },
33
+
34
+ })
35
+ </script>
36
+ <style lang="scss">
37
+ @import '../../../sass/_variables.scss';
38
+ .octopus-app{
39
+ .player-container {
40
+ .timeline-button {
41
+ background: black;
42
+ padding: 0.1rem;
43
+ border-radius: 50%;
44
+ width: 70px;
45
+ height: 70px;
46
+ font-size: 0.7rem;
47
+ font-weight: bold;
48
+ justify-content: center;
49
+ display: flex;
50
+ flex-direction: column;
51
+ align-items: center;
52
+ cursor: pointer;
53
+ color: $octopus-primary-color;
54
+ margin-left: 0.5rem;
55
+ @media (max-width: 960px) {
56
+ display: none;
57
+ }
58
+ }
59
+ }
60
+ }
61
+ </style>
@@ -1,23 +1,25 @@
1
- export const selenium = {
1
+
2
+ export const selenium ={
2
3
  methods: {
3
- seleniumFormat(string: string) {
4
+ seleniumFormat(string: string): string {
4
5
  return string.toLowerCase().replace(/\s/g, '');
5
6
  },
6
7
  },
7
8
  };
8
- export const cookies = {
9
+ export const cookies =
10
+ {
9
11
  methods: {
10
- setCookie(name: string, value: any) {
11
- var date = new Date();
12
+ setCookie(name: string, value: string, domain= ""): void {
13
+ const date = new Date();
12
14
  date.setTime(date.getTime() + 24 * 60 * 60 * 1000);
13
- var expires = '; expires=' + date.toUTCString();
14
- document.cookie = name + '=' + (value || '') + expires + '; path=/';
15
+ const expires = '; expires=' + date.toUTCString();
16
+ document.cookie = name + '=' + (value || '') + expires +domain+ '; path=/';
15
17
  },
16
- getCookie(name: string) {
17
- var nameEQ = name + '=';
18
- var ca = document.cookie.split(';');
19
- for (var i = 0; i < ca.length; i++) {
20
- var c = ca[i];
18
+ getCookie(name: string): string|null {
19
+ const nameEQ = name + '=';
20
+ const ca = document.cookie.split(';');
21
+ for (let i = 0; i < ca.length; i++) {
22
+ let c = ca[i];
21
23
  while (c.charAt(0) == ' ') c = c.substring(1, c.length);
22
24
  if (0 === c.indexOf(nameEQ))
23
25
  return c.substring(nameEQ.length, c.length);
@@ -27,21 +29,21 @@ export const cookies = {
27
29
  },
28
30
  };
29
31
 
30
- export const displayMethods = {
32
+ export const displayMethods ={
31
33
  methods: {
32
- urlify(text: string) {
33
- let urlRegex = /(https?:\/\/[^\s]+)/g;
34
+ urlify(text: string): string {
35
+ const urlRegex = /(https?:\/\/[^\s<]+)/g;
34
36
  if (!text) return '';
35
37
  return text.replace(urlRegex, (url: string) => {
36
- return '<a href="' + url + '" target="_blank">' + url + '</a>';
38
+ return '<a href="' + url + '" target="_blank" rel="noopener">' + url + '</a>';
37
39
  });
38
40
  },
39
- async onCopyCode(link: string, callback: ()=>void) {
41
+ async onCopyCode(link: string, callback: () => void): Promise<void> {
40
42
  if ('undefined' !== typeof navigator.clipboard) {
41
43
  await navigator.clipboard.writeText(link);
42
44
  return callback();
43
45
  }
44
- let textArea = document.createElement('textarea');
46
+ const textArea = document.createElement('textarea');
45
47
  textArea.value = link;
46
48
  textArea.style.position = 'fixed';
47
49
  document.body.appendChild(textArea);
@@ -53,3 +55,4 @@ export const displayMethods = {
53
55
  },
54
56
  },
55
57
  };
58
+
@@ -0,0 +1,17 @@
1
+ import { AxiosError } from 'axios';
2
+ import { defineComponent } from 'vue';
3
+ export const handle403 = defineComponent({
4
+ methods: {
5
+ handle403(error: AxiosError): void {
6
+ if (403 === error.response?.status) {
7
+ if(!this.$store.state.authentication.isAuthenticated){
8
+ window.location.href = window.location.origin + "/sso/login";
9
+ }else{
10
+ this.$router.push({
11
+ path: '/main/pub/error'
12
+ });
13
+ }
14
+ }
15
+ },
16
+ },
17
+ });
@@ -0,0 +1,24 @@
1
+
2
+ import { Category } from "@/store/class/general/category";
3
+ import { orgaFilter } from '../mixins/organisationFilter';
4
+ import octopusApi from '@saooti/octopus-api';
5
+ import { state } from '../../store/paramStore';
6
+ import { defineComponent } from 'vue';
7
+ export const initSDK = defineComponent({
8
+ mixins: [orgaFilter],
9
+ methods: {
10
+ async initSdk() {
11
+ if (0 === (state.generalParameters.allCategories as Array<Category>).length) {
12
+ octopusApi.fetchDataWithParams<Array<Category>>(0, `iab/list${state.octopusApi.organisationId? '/'+state.octopusApi.organisationId : ''}`, { lang: this.$i18n.locale }).then((data: Array<Category>) => {
13
+ this.$store.commit('categoriesSet', data);
14
+ });
15
+ }else{
16
+ this.$store.commit('categoriesSet', (state.generalParameters.allCategories as Array<Category>));
17
+ }
18
+ const captcha = (document.getElementsByClassName('grecaptcha-badge')[0] as HTMLElement);
19
+ if (captcha) {
20
+ captcha.style.display = 'none';
21
+ }
22
+ },
23
+ },
24
+ });
@@ -0,0 +1,15 @@
1
+ import { state } from '../../store/paramStore';
2
+ import { defineComponent } from 'vue';
3
+ export const orgaComputed = defineComponent({
4
+ computed: {
5
+ myOrganisationId(): string|undefined {
6
+ return state.generalParameters.organisationId;
7
+ },
8
+ authenticated(): boolean {
9
+ return (state.generalParameters.authenticated as boolean);
10
+ },
11
+ filterOrga(): string {
12
+ return this.$store.state.filter.organisationId;
13
+ },
14
+ },
15
+ });
@@ -0,0 +1,41 @@
1
+
2
+ import { handle403 } from '../mixins/handle403';
3
+ import { Rubriquage } from "@/store/class/rubrique/rubriquage";
4
+ import octopusApi from '@saooti/octopus-api';
5
+ import { defineComponent } from 'vue'
6
+ import { AxiosError } from 'axios';
7
+ import { Organisation } from '@/store/class/general/organisation';
8
+ export const orgaFilter = defineComponent({
9
+ mixins: [handle403],
10
+ methods: {
11
+ async selectOrganisation(organisationId: string): Promise<void> {
12
+ try {
13
+ const response = await octopusApi.fetchData<Organisation>(0,`organisation/${organisationId}`);
14
+ const data = await octopusApi.fetchDataWithParams<Array<Rubriquage>>(0, 'rubriquage/find/'+organisationId,{
15
+ sort:'HOMEPAGEORDER',
16
+ homePageOrder: true
17
+ }, true);
18
+ const isLive = await octopusApi.fetchData<boolean>(0, 'organisation/liveEnabled/'+organisationId);
19
+ this.$store.commit('filterOrga', {
20
+ orgaId: organisationId,
21
+ imgUrl: response.imageUrl,
22
+ name: response.name,
23
+ rubriquageArray: data.filter((element: Rubriquage)=>{
24
+ return element.rubriques.length;
25
+ }),
26
+ isLive: isLive
27
+ });
28
+ const queries = this.$route.query;
29
+ this.$router.replace({ query: {...queries, ...{productor: organisationId} } });
30
+ } catch (error) {
31
+ this.handle403((error as AxiosError));
32
+ }
33
+ },
34
+ removeSelectedOrga(): void{
35
+ if (this.$route.query.productor) {
36
+ this.$router.push({ query: {...this.$route.query, ...{productor: undefined} } });
37
+ }
38
+ this.$store.commit('filterOrga', { orgaId: undefined });
39
+ }
40
+ },
41
+ });
@@ -0,0 +1,98 @@
1
+
2
+ import { mapState } from 'vuex';
3
+ import { state } from '../../../store/paramStore';
4
+ import octopusApi from '@saooti/octopus-api';
5
+ import { CommentPodcast } from '@/store/class/general/comment';
6
+ import { StoreState } from '@/store/typeAppStore';
7
+ import { defineComponent } from 'vue';
8
+ import { FetchParam } from '@/store/class/general/fetchParam';
9
+ import { InterfacePageable } from '@/store/class/general/interfacePageable';
10
+ export const playerComment = defineComponent({
11
+ data() {
12
+ return {
13
+ comments: [] as Array<CommentPodcast>,
14
+ };
15
+ },
16
+ computed: {
17
+ ...mapState({
18
+ commentsLoaded(state: StoreState){ return state.comments.loadedComments},
19
+ live(state: StoreState) { return state.player.live},
20
+ podcast(state: StoreState) { return state.player.podcast}
21
+ }),
22
+ organisationId(): string|undefined {
23
+ return state.generalParameters.organisationId;
24
+ },
25
+ },
26
+
27
+ watch: {
28
+ commentsLoaded(): void {
29
+ this.initComments(true);
30
+ },
31
+ },
32
+ methods: {
33
+ editRight(organisation: string): boolean {
34
+ if (
35
+ (state.generalParameters.isCommments &&
36
+ this.organisationId === organisation) ||
37
+ state.generalParameters.isAdmin
38
+ )
39
+ return true;
40
+ return false;
41
+ },
42
+ async initComments(refresh = false): Promise<void> {
43
+ let podcastId, organisation;
44
+ if (this.podcast) {
45
+ podcastId = this.podcast.podcastId;
46
+ organisation = this.podcast.organisation.id;
47
+ } else if (this.live) {
48
+ podcastId = this.live.livePodcastId;
49
+ organisation = this.live.organisation.id;
50
+ }
51
+ if (
52
+ refresh &&
53
+ podcastId &&
54
+ this.$store.state.comments.actualPodcastId !== podcastId
55
+ ) {
56
+ return;
57
+ }
58
+ let first = 0;
59
+ let count = 0;
60
+ const size = 50;
61
+ if (
62
+ podcastId &&
63
+ this.$store.state.comments.actualPodcastId === podcastId
64
+ ) {
65
+ this.comments = this.commentsLoaded;
66
+ if (
67
+ this.commentsLoaded &&
68
+ this.commentsLoaded.length < this.$store.state.comments.totalCount
69
+ ) {
70
+ first = this.commentsLoaded.length;
71
+ count = this.$store.state.comments.totalCount;
72
+ }
73
+ }
74
+ if (
75
+ (!podcastId ||
76
+ this.$store.state.comments.actualPodcastId === podcastId) &&
77
+ 0 === first
78
+ )
79
+ return;
80
+ while (0 === first || this.comments.length < count) {
81
+ const param: FetchParam = {
82
+ first: first,
83
+ size: size,
84
+ podcastId: podcastId,
85
+ };
86
+ if (!this.editRight(organisation? organisation : '')) {
87
+ param.status = ['Valid'];
88
+ }
89
+ const data = await octopusApi.postDataPublic<InterfacePageable<CommentPodcast>>(2, 'getRootCom',param);
90
+ first += size;
91
+ count = data.totalElements;
92
+ this.comments = this.comments.concat(data.content).filter((c: CommentPodcast) => {
93
+ return null !== c;
94
+ });
95
+ }
96
+ },
97
+ },
98
+ })
@@ -0,0 +1,113 @@
1
+ import { Podcast } from '@/store/class/general/podcast';
2
+ import DurationHelper from '../../../helper/duration';
3
+ import { state } from '../../../store/paramStore';
4
+ import { defineComponent } from 'vue';
5
+ import { RouteLocationRaw } from 'vue-router';
6
+ export const playerDisplay = defineComponent({
7
+ props: {
8
+ hlsReady: { default: false , type: Boolean},
9
+ },
10
+ computed:{
11
+ playedTime(): string{
12
+ if (this.$store.state.player.elapsed && this.$store.state.player.elapsed > 0 && this.$store.state.player.total && this.$store.state.player.total > 0) {
13
+ return DurationHelper.formatDuration(
14
+ Math.round(this.$store.state.player.elapsed * this.$store.state.player.total)
15
+ );
16
+ }
17
+ return '--:--';
18
+ },
19
+ totalTime(): string {
20
+ if (this.$store.state.player.elapsed && this.$store.state.player.elapsed > 0 && this.$store.state.player.total && this.$store.state.player.total > 0)
21
+ return DurationHelper.formatDuration(Math.round(this.$store.state.player.total));
22
+ return '--:--';
23
+ },
24
+ isPlaying(): boolean {
25
+ return 'PLAYING' === this.$store.state.player.status;
26
+ },
27
+ isPaused(): boolean {
28
+ return 'PAUSED' === this.$store.state.player.status;
29
+ },
30
+ podcast(): undefined|Podcast{
31
+ return this.$store.state.player.podcast;
32
+ },
33
+ isImage(): boolean {
34
+ return (state.player.image as boolean);
35
+ },
36
+ podcastImage(): string{
37
+ if (this.$store.state.player.podcast) return this.$store.state.player.podcast.imageUrl;
38
+ return '';
39
+ },
40
+ podcastShareUrl(): RouteLocationRaw|string {
41
+ if (this.podcast) {
42
+ return {
43
+ name: 'podcast',
44
+ params: { podcastId: this.podcast.podcastId.toString() },
45
+ query: { productor: this.$store.state.filter.organisationId },
46
+ };
47
+ }
48
+ return '';
49
+ },
50
+ isEmissionName(): boolean {
51
+ return (state.player.emissionName as boolean);
52
+ },
53
+ podcastTitle(): string {
54
+ if (this.$store.state.player.podcast) {
55
+ if (this.isEmissionName)
56
+ return this.emissionName + ' - ' + this.$store.state.player.podcast.title;
57
+ return this.$store.state.player.podcast.title;
58
+ }
59
+ if (this.$store.state.player.media) return this.$store.state.player.media.title;
60
+ if (this.$store.state.player.live) {
61
+ if (!this.hlsReady)
62
+ return this.$store.state.player.live.title + ' (' + this.$t('Start in a while') + ')';
63
+ return this.$store.state.player.live.title;
64
+ }
65
+ return '';
66
+ },
67
+ emissionName(): string {
68
+ if (this.$store.state.player.podcast) return this.$store.state.player.podcast.emission.name;
69
+ return '';
70
+ },
71
+ },
72
+ created(){
73
+ window.addEventListener('keydown', this.addKeyboardControl);
74
+ },
75
+ beforeUnmount() {
76
+ window.removeEventListener('keydown', this.addKeyboardControl);
77
+ },
78
+
79
+ methods: {
80
+ addKeyboardControl(event: KeyboardEvent): void{
81
+ if(!event || null ===event){return;}
82
+ const element = event.target as HTMLElement;
83
+ if (!element || 'INPUT' == element.tagName.toUpperCase() || 'TEXTAREA' == element.tagName.toUpperCase()){return;}
84
+ if (' ' === event.key || 'Spacebar' === event.key) {
85
+ event.preventDefault();
86
+ this.switchPausePlay();
87
+ }else if ('ArrowRight' === event.key && event.ctrlKey) {
88
+ const audioPlayer: HTMLAudioElement|null = document.querySelector('#audio-player');
89
+ if(!audioPlayer){return;}
90
+ audioPlayer.currentTime += 15;
91
+ }else if ('ArrowLeft' === event.key && event.ctrlKey) {
92
+ const audioPlayer: HTMLAudioElement|null = document.querySelector('#audio-player');
93
+ if(!audioPlayer){return;}
94
+ audioPlayer.currentTime -=15;
95
+ }
96
+ },
97
+ switchPausePlay(): void {
98
+ const audioPlayer: HTMLAudioElement|null = document.querySelector('#audio-player');
99
+ if(!audioPlayer){return;}
100
+ if (audioPlayer.paused) {
101
+ this.onPlay();
102
+ } else {
103
+ this.onPause();
104
+ }
105
+ },
106
+ onPlay(): void {
107
+ this.$store.commit('playerPause', false);
108
+ },
109
+ onPause(): void {
110
+ this.$store.commit('playerPause', true);
111
+ },
112
+ }
113
+ });
@@ -0,0 +1,111 @@
1
+ import { mapState } from 'vuex';
2
+ import { state } from '../../../store/paramStore';
3
+ import octopusApi from '@saooti/octopus-api';
4
+ /* eslint-disable */
5
+ let Hls:any = null;
6
+ /* eslint-enable */
7
+ import { StoreState } from '@/store/typeAppStore';
8
+ import { defineComponent } from 'vue';
9
+ export const playerLive = defineComponent({
10
+ data() {
11
+ return {
12
+ listenTime: 0 as number,
13
+ notListenTime: 0 as number,
14
+ lastSend: 0 as number,
15
+ hlsReady: false as boolean,
16
+ downloadId: null as string|null,
17
+ };
18
+ },
19
+ computed: {
20
+ ...mapState({
21
+ live(state: StoreState) { return state.player.live}
22
+ }),
23
+ },
24
+ methods: {
25
+ onPlay(): void {
26
+ this.$store.commit('playerPause', false);
27
+ },
28
+ async playLive(): Promise<void> {
29
+ if (!this.live) return;
30
+ const hlsStreamUrl =
31
+ state.podcastPage.hlsUri +
32
+ 'stream/dev.' +
33
+ this.live.conferenceId +
34
+ '/index.m3u8';
35
+ try {
36
+ await this.initHls(hlsStreamUrl);
37
+ } catch (error) {
38
+ console.log(error);
39
+ setTimeout(() => {
40
+ this.playLive();
41
+ }, 1000);
42
+ }
43
+ },
44
+ async initHls(hlsStreamUrl: string): Promise<void> {
45
+ return new Promise<void>(async(resolve, reject) => {
46
+ if(null === Hls){
47
+ //TODO -> Version light min quand ce sera possible
48
+ await import('hls.js/dist/hls.js').then((hlsLibrary) => {
49
+ Hls = hlsLibrary.default;
50
+ })
51
+ await import('hls.js').then((hlsLibrary) => {
52
+ Hls = hlsLibrary.default;
53
+ })
54
+ }
55
+ if (!Hls.isSupported()) {
56
+ reject('Hls is not supported ! ');
57
+ }
58
+ const hls = new Hls();
59
+ /* if(this.$store.state.authentication.isAuthenticated && this.$store.state.oAuthParam.accessToken){
60
+ hls = new Hls({xhrSetup:
61
+ (xhr: XMLHttpRequest) => {
62
+ xhr.setRequestHeader("Authorization", "Bearer " + this.$store.state.oAuthParam.accessToken);
63
+ }
64
+ }
65
+ );
66
+ } */
67
+ hls.on(Hls.Events.MANIFEST_PARSED, async () => {
68
+ if(!this.live){ return; }
69
+ let downloadId = null;
70
+ try {
71
+ downloadId = await octopusApi.putDataPublic<string | null>(0, 'podcast/prepare/live/'+this.live.livePodcastId, undefined);
72
+ await octopusApi.fetchDataPublicWithParams<string | null>(0,'podcast/download/live/' + this.live.livePodcastId+".m3u8",{
73
+ 'downloadId': null!==downloadId ? downloadId : undefined,
74
+ 'origin':'octopus',
75
+ 'distributorId':this.$store.state.authentication.organisationId
76
+ });
77
+ this.setDownloadId(downloadId);
78
+ } catch (error) {
79
+ console.log('ERROR downloadId');
80
+ }
81
+ this.hlsReady = true;
82
+ const audio: HTMLElement|null = document.getElementById('audio-player');
83
+ hls.attachMedia((audio as HTMLAudioElement));
84
+ await (audio as HTMLAudioElement).play();
85
+ this.onPlay();
86
+ resolve();
87
+ });
88
+ hls.on(Hls.Events.ERROR, async() => {
89
+ reject('There is an error while reading media content');
90
+ });
91
+ hls.loadSource(hlsStreamUrl);
92
+ });
93
+ },
94
+ setDownloadId(newValue: string|null): void {
95
+ this.endListeningProgress();
96
+ this.downloadId = newValue;
97
+ },
98
+ async endListeningProgress(): Promise<void> {
99
+ if (!this.downloadId) return;
100
+ try {
101
+ await octopusApi.putDataPublic(0, 'podcast/listen/' + this.downloadId + '?seconds=' + Math.round(this.listenTime), undefined);
102
+ } catch{
103
+ //Do nothing
104
+ }
105
+ this.downloadId = null;
106
+ this.notListenTime = 0;
107
+ this.lastSend = 0;
108
+ this.listenTime = 0;
109
+ },
110
+ },
111
+ })