Reorganize client shared modules
[oweals/peertube.git] / client / src / app / shared / shared-moderation / video-report.component.ts
1 import { mapValues, pickBy } from 'lodash-es'
2 import { buildVideoEmbed, buildVideoLink } from 'src/assets/player/utils'
3 import { Component, Input, OnInit, ViewChild } from '@angular/core'
4 import { DomSanitizer, SafeHtml } from '@angular/platform-browser'
5 import { Notifier } from '@app/core'
6 import { FormReactive, FormValidatorService, VideoAbuseValidatorsService } from '@app/shared/shared-forms'
7 import { NgbModal } from '@ng-bootstrap/ng-bootstrap'
8 import { NgbModalRef } from '@ng-bootstrap/ng-bootstrap/modal/modal-ref'
9 import { I18n } from '@ngx-translate/i18n-polyfill'
10 import { videoAbusePredefinedReasonsMap, VideoAbusePredefinedReasonsString } from '@shared/models/videos/abuse/video-abuse-reason.model'
11 import { Video } from '../shared-main'
12 import { VideoAbuseService } from './video-abuse.service'
13
14 @Component({
15   selector: 'my-video-report',
16   templateUrl: './video-report.component.html',
17   styleUrls: [ './video-report.component.scss' ]
18 })
19 export class VideoReportComponent extends FormReactive implements OnInit {
20   @Input() video: Video = null
21
22   @ViewChild('modal', { static: true }) modal: NgbModal
23
24   error: string = null
25   predefinedReasons: { id: VideoAbusePredefinedReasonsString, label: string, description?: string, help?: string }[] = []
26   embedHtml: SafeHtml
27
28   private openedModal: NgbModalRef
29
30   constructor (
31     protected formValidatorService: FormValidatorService,
32     private modalService: NgbModal,
33     private videoAbuseValidatorsService: VideoAbuseValidatorsService,
34     private videoAbuseService: VideoAbuseService,
35     private notifier: Notifier,
36     private sanitizer: DomSanitizer,
37     private i18n: I18n
38   ) {
39     super()
40   }
41
42   get currentHost () {
43     return window.location.host
44   }
45
46   get originHost () {
47     if (this.isRemoteVideo()) {
48       return this.video.account.host
49     }
50
51     return ''
52   }
53
54   get timestamp () {
55     return this.form.get('timestamp').value
56   }
57
58   getVideoEmbed () {
59     return this.sanitizer.bypassSecurityTrustHtml(
60       buildVideoEmbed(
61         buildVideoLink({
62           baseUrl: this.video.embedUrl,
63           title: false,
64           warningTitle: false
65         })
66       )
67     )
68   }
69
70   ngOnInit () {
71     this.buildForm({
72       reason: this.videoAbuseValidatorsService.VIDEO_ABUSE_REASON,
73       predefinedReasons: mapValues(videoAbusePredefinedReasonsMap, r => null),
74       timestamp: {
75         hasStart: null,
76         startAt: null,
77         hasEnd: null,
78         endAt: null
79       }
80     })
81
82     this.predefinedReasons = [
83       {
84         id: 'violentOrRepulsive',
85         label: this.i18n('Violent or repulsive'),
86         help: this.i18n('Contains offensive, violent, or coarse language or iconography.')
87       },
88       {
89         id: 'hatefulOrAbusive',
90         label: this.i18n('Hateful or abusive'),
91         help: this.i18n('Contains abusive, racist or sexist language or iconography.')
92       },
93       {
94         id: 'spamOrMisleading',
95         label: this.i18n('Spam, ad or false news'),
96         help: this.i18n('Contains marketing, spam, purposefully deceitful news, or otherwise misleading thumbnail/text/tags. Please provide reputable sources to report hoaxes.')
97       },
98       {
99         id: 'privacy',
100         label: this.i18n('Privacy breach or doxxing'),
101         help: this.i18n('Contains personal information that could be used to track, identify, contact or impersonate someone (e.g. name, address, phone number, email, or credit card details).')
102       },
103       {
104         id: 'rights',
105         label: this.i18n('Intellectual property violation'),
106         help: this.i18n('Infringes my intellectual property or copyright, wrt. the regional rules with which the server must comply.')
107       },
108       {
109         id: 'serverRules',
110         label: this.i18n('Breaks server rules'),
111         description: this.i18n('Anything not included in the above that breaks the terms of service, code of conduct, or general rules in place on the server.')
112       },
113       {
114         id: 'thumbnails',
115         label: this.i18n('Thumbnails'),
116         help: this.i18n('The above can only be seen in thumbnails.')
117       },
118       {
119         id: 'captions',
120         label: this.i18n('Captions'),
121         help: this.i18n('The above can only be seen in captions (please describe which).')
122       }
123     ]
124
125     this.embedHtml = this.getVideoEmbed()
126   }
127
128   show () {
129     this.openedModal = this.modalService.open(this.modal, { centered: true, keyboard: false, size: 'lg' })
130   }
131
132   hide () {
133     this.openedModal.close()
134     this.openedModal = null
135   }
136
137   report () {
138     const reason = this.form.get('reason').value
139     const predefinedReasons = Object.keys(pickBy(this.form.get('predefinedReasons').value)) as VideoAbusePredefinedReasonsString[]
140     const { hasStart, startAt, hasEnd, endAt } = this.form.get('timestamp').value
141
142     this.videoAbuseService.reportVideo({
143       id: this.video.id,
144       reason,
145       predefinedReasons,
146       startAt: hasStart && startAt ? startAt : undefined,
147       endAt: hasEnd && endAt ? endAt : undefined
148     }).subscribe(
149       () => {
150         this.notifier.success(this.i18n('Video reported.'))
151         this.hide()
152       },
153
154       err => this.notifier.error(err.message)
155     )
156   }
157
158   isRemoteVideo () {
159     return !this.video.isLocal
160   }
161 }