]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - client/src/app/+admin/overview/videos/video-list.component.ts
Support ICU in TS components
[github/Chocobozzz/PeerTube.git] / client / src / app / +admin / overview / videos / video-list.component.ts
CommitLineData
33f6dce1 1import { SortMeta } from 'primeng/api'
61f85385 2import { finalize } from 'rxjs/operators'
3cfa8176 3import { Component, OnInit, ViewChild } from '@angular/core'
33f6dce1
C
4import { ActivatedRoute, Router } from '@angular/router'
5import { AuthService, ConfirmService, Notifier, RestPagination, RestTable } from '@app/core'
eaa52952 6import { prepareIcu } from '@app/helpers'
33f6dce1 7import { AdvancedInputFilter } from '@app/shared/shared-forms'
61f85385 8import { DropdownAction, Video, VideoService } from '@app/shared/shared-main'
3cfa8176 9import { VideoBlockComponent, VideoBlockService } from '@app/shared/shared-moderation'
33f6dce1 10import { VideoActionsDisplayType } from '@app/shared/shared-video-miniature'
61f85385 11import { UserRight, VideoPrivacy, VideoState, VideoStreamingPlaylistType } from '@shared/models'
05ac4ac7 12import { VideoAdminService } from './video-admin.service'
33f6dce1
C
13
14@Component({
15 selector: 'my-video-list',
16 templateUrl: './video-list.component.html',
17 styleUrls: [ './video-list.component.scss' ]
18})
19export class VideoListComponent extends RestTable implements OnInit {
3cfa8176
C
20 @ViewChild('videoBlockModal') videoBlockModal: VideoBlockComponent
21
33f6dce1
C
22 videos: Video[] = []
23
24 totalRecords = 0
7e7d8e48 25 sort: SortMeta = { field: 'publishedAt', order: -1 }
33f6dce1
C
26 pagination: RestPagination = { count: this.rowsPerPage, start: 0 }
27
28 bulkVideoActions: DropdownAction<Video[]>[][] = []
29
30 selectedVideos: Video[] = []
31
231ff4af 32 inputFilters: AdvancedInputFilter[]
33f6dce1
C
33
34 videoActionsOptions: VideoActionsDisplayType = {
35 playlist: false,
36 download: false,
37 update: true,
38 blacklist: true,
39 delete: true,
40 report: false,
41 duplicate: true,
42 mute: true,
b46cf4b9 43 liveInfo: false,
ad5db104 44 removeFiles: true,
384ba8b7
C
45 transcoding: true,
46 studio: true,
47 stats: true
33f6dce1
C
48 }
49
231ff4af 50 loading = true
61f85385 51
33f6dce1
C
52 constructor (
53 protected route: ActivatedRoute,
54 protected router: Router,
55 private confirmService: ConfirmService,
56 private auth: AuthService,
57 private notifier: Notifier,
05ac4ac7 58 private videoService: VideoService,
3cfa8176
C
59 private videoAdminService: VideoAdminService,
60 private videoBlockService: VideoBlockService
33f6dce1
C
61 ) {
62 super()
63 }
64
65 get authUser () {
66 return this.auth.getUser()
67 }
68
69 ngOnInit () {
70 this.initialize()
71
05ac4ac7 72 this.inputFilters = this.videoAdminService.buildAdminInputFilter()
231ff4af 73
33f6dce1
C
74 this.bulkVideoActions = [
75 [
76 {
77 label: $localize`Delete`,
78 handler: videos => this.removeVideos(videos),
b46cf4b9
C
79 isDisplayed: () => this.authUser.hasRight(UserRight.REMOVE_ANY_VIDEO),
80 iconName: 'delete'
3cfa8176
C
81 },
82 {
83 label: $localize`Block`,
84 handler: videos => this.videoBlockModal.show(videos),
b46cf4b9
C
85 isDisplayed: videos => this.authUser.hasRight(UserRight.MANAGE_VIDEO_BLACKLIST) && videos.every(v => !v.blacklisted),
86 iconName: 'no'
3cfa8176
C
87 },
88 {
89 label: $localize`Unblock`,
90 handler: videos => this.unblockVideos(videos),
b46cf4b9
C
91 isDisplayed: videos => this.authUser.hasRight(UserRight.MANAGE_VIDEO_BLACKLIST) && videos.every(v => v.blacklisted),
92 iconName: 'undo'
93 }
94 ],
95 [
ad5db104
C
96 {
97 label: $localize`Run HLS transcoding`,
98 handler: videos => this.runTranscoding(videos, 'hls'),
99 isDisplayed: videos => videos.every(v => v.canRunTranscoding(this.authUser)),
100 iconName: 'cog'
101 },
102 {
103 label: $localize`Run WebTorrent transcoding`,
104 handler: videos => this.runTranscoding(videos, 'webtorrent'),
105 isDisplayed: videos => videos.every(v => v.canRunTranscoding(this.authUser)),
106 iconName: 'cog'
107 },
b46cf4b9
C
108 {
109 label: $localize`Delete HLS files`,
110 handler: videos => this.removeVideoFiles(videos, 'hls'),
ad5db104 111 isDisplayed: videos => videos.every(v => v.canRemoveFiles(this.authUser)),
b46cf4b9
C
112 iconName: 'delete'
113 },
114 {
115 label: $localize`Delete WebTorrent files`,
116 handler: videos => this.removeVideoFiles(videos, 'webtorrent'),
ad5db104 117 isDisplayed: videos => videos.every(v => v.canRemoveFiles(this.authUser)),
b46cf4b9 118 iconName: 'delete'
33f6dce1
C
119 }
120 ]
121 ]
122 }
123
124 getIdentifier () {
125 return 'VideoListComponent'
126 }
127
128 isInSelectionMode () {
129 return this.selectedVideos.length !== 0
130 }
131
d324756e 132 getPrivacyBadgeClass (video: Video) {
dd6d2a7c 133 if (video.privacy.id === VideoPrivacy.PUBLIC) return 'badge-green'
2760b454
C
134
135 return 'badge-yellow'
136 }
137
d324756e
C
138 isUnpublished (video: Video) {
139 return video.state.id !== VideoState.LIVE_ENDED && video.state.id !== VideoState.PUBLISHED
2760b454
C
140 }
141
142 isAccountBlocked (video: Video) {
143 return video.blockedOwner
144 }
145
146 isServerBlocked (video: Video) {
147 return video.blockedServer
148 }
149
150 isVideoBlocked (video: Video) {
151 return video.blacklisted
152 }
153
d324756e
C
154 isImport (video: Video) {
155 return video.state.id === VideoState.TO_IMPORT
156 }
157
3c10840f 158 isHLS (video: Video) {
d5d9c5b7
C
159 const p = video.streamingPlaylists.find(p => p.type === VideoStreamingPlaylistType.HLS)
160 if (!p) return false
161
162 return p.files.length !== 0
3c10840f
C
163 }
164
165 isWebTorrent (video: Video) {
166 return video.files.length !== 0
167 }
168
169 getFilesSize (video: Video) {
170 let files = video.files
171
172 if (this.isHLS(video)) {
173 files = files.concat(video.streamingPlaylists[0].files)
174 }
175
176 return files.reduce((p, f) => p += f.size, 0)
177 }
178
b46cf4b9 179 reloadData () {
33f6dce1
C
180 this.selectedVideos = []
181
61f85385
C
182 this.loading = true
183
05ac4ac7 184 this.videoAdminService.getAdminVideos({
33f6dce1
C
185 pagination: this.pagination,
186 sort: this.sort,
187 search: this.search
61f85385
C
188 }).pipe(finalize(() => this.loading = false))
189 .subscribe({
190 next: resultList => {
191 this.videos = resultList.data
192 this.totalRecords = resultList.total
193 },
194
195 error: err => this.notifier.error(err.message)
196 })
33f6dce1
C
197 }
198
199 private async removeVideos (videos: Video[]) {
eaa52952
C
200 const message = prepareIcu($localize`Are you sure you want to delete {count, plural, =1 {this video} other {these {count} videos}}?`)(
201 { count: videos.length },
202 $localize`Are you sure you want to delete these ${videos.length} videos?`
203 )
204
33f6dce1
C
205 const res = await this.confirmService.confirm(message, $localize`Delete`)
206 if (res === false) return
207
208 this.videoService.removeVideo(videos.map(v => v.id))
209 .subscribe({
210 next: () => {
eaa52952
C
211 this.notifier.success(
212 prepareIcu($localize`Deleted {count, plural, =1 {1 video} other {{count} videos}}.`)(
213 { count: videos.length },
214 $localize`Deleted ${videos.length} videos.`
215 )
216 )
217
3cfa8176
C
218 this.reloadData()
219 },
220
221 error: err => this.notifier.error(err.message)
222 })
223 }
224
225 private unblockVideos (videos: Video[]) {
226 this.videoBlockService.unblockVideo(videos.map(v => v.id))
227 .subscribe({
228 next: () => {
eaa52952
C
229 this.notifier.success(
230 prepareIcu($localize`Unblocked {count, plural, =1 {1 video} other {{count} videos}}.`)(
231 { count: videos.length },
232 $localize`Unblocked ${videos.length} videos.`
233 )
234 )
235
33f6dce1
C
236 this.reloadData()
237 },
238
239 error: err => this.notifier.error(err.message)
240 })
241 }
b46cf4b9
C
242
243 private async removeVideoFiles (videos: Video[], type: 'hls' | 'webtorrent') {
eaa52952
C
244 let message: string
245
246 if (type === 'hls') {
247 // eslint-disable-next-line max-len
248 message = prepareIcu($localize`Are you sure you want to delete {count, plural, =1 {1 HLS streaming playlist} other {{count} HLS streaming playlists}}?`)(
249 { count: videos.length },
250 $localize`Are you sure you want to delete ${videos.length} HLS streaming playlists?`
251 )
252 } else {
253 // eslint-disable-next-line max-len
254 message = prepareIcu($localize`Are you sure you want to delete WebTorrent files of {count, plural, =1 {1 video} other {{count} videos}}?`)(
255 { count: videos.length },
256 $localize`Are you sure you want to delete WebTorrent files of ${videos.length} videos?`
257 )
258 }
b46cf4b9
C
259
260 const res = await this.confirmService.confirm(message, $localize`Delete`)
261 if (res === false) return
262
263 this.videoService.removeVideoFiles(videos.map(v => v.id), type)
264 .subscribe({
265 next: () => {
266 this.notifier.success($localize`Files were removed.`)
267 this.reloadData()
268 },
269
270 error: err => this.notifier.error(err.message)
271 })
272 }
ad5db104
C
273
274 private runTranscoding (videos: Video[], type: 'hls' | 'webtorrent') {
275 this.videoService.runTranscoding(videos.map(v => v.id), type)
276 .subscribe({
277 next: () => {
278 this.notifier.success($localize`Transcoding jobs created.`)
279
280 this.reloadData()
281 },
282
283 error: err => this.notifier.error(err.message)
284 })
285 }
33f6dce1 286}