import { AuthUser } from '@app/core'
import { User } from '@app/core/users/user.model'
-import { durationToString, getAbsoluteAPIUrl, getAbsoluteEmbedUrl } from '@app/helpers'
-import { Account } from '@app/shared/shared-main/account/account.model'
+import { durationToString, prepareIcu, getAbsoluteAPIUrl, getAbsoluteEmbedUrl } from '@app/helpers'
import { Actor } from '@app/shared/shared-main/account/actor.model'
-import { VideoChannel } from '@app/shared/shared-main/video-channel/video-channel.model'
+import { buildVideoWatchPath } from '@shared/core-utils'
import { peertubeTranslate } from '@shared/core-utils/i18n'
import {
ActorImage,
- ServerConfig,
+ HTMLServerConfig,
UserRight,
Video as VideoServerModel,
VideoConstant,
+ VideoFile,
VideoPrivacy,
VideoScheduleUpdate,
- VideoState
+ VideoState,
+ VideoStreamingPlaylist,
+ VideoStreamingPlaylistType
} from '@shared/models'
export class Video implements VideoServerModel {
+ private static readonly viewsICU = prepareIcu($localize`{views, plural, =0 {No view} =1 {1 view} other {{views} views}}`)
+ private static readonly viewersICU = prepareIcu($localize`{viewers, plural, =0 {No viewers} =1 {1 viewer} other {{viewers} viewers}}`)
+
byVideoChannel: string
byAccount: string
- videoChannelAvatarUrl: string
-
createdAt: Date
updatedAt: Date
publishedAt: Date
licence: VideoConstant<number>
language: VideoConstant<string>
privacy: VideoConstant<VideoPrivacy>
+
description: string
+
duration: number
durationLabel: string
+
id: number
uuid: string
+ shortUUID: string
+
isLocal: boolean
+
name: string
serverHost: string
thumbnailPath: string
embedPath: string
embedUrl: string
- url?: string
+ url: string
views: number
+ viewers: number
+
likes: number
dislikes: number
nsfw: boolean
waitTranscoding?: boolean
state?: VideoConstant<VideoState>
scheduledUpdate?: VideoScheduleUpdate
+
blacklisted?: boolean
- blockedReason?: string
+ blacklistedReason?: string
+
+ blockedOwner?: boolean
+ blockedServer?: boolean
account: {
id: number
displayName: string
url: string
host: string
- avatar?: ActorImage
+
+ // TODO: remove, deprecated in 4.2
+ avatar: ActorImage
+
+ avatars: ActorImage[]
}
channel: {
displayName: string
url: string
host: string
- avatar?: ActorImage
+
+ // TODO: remove, deprecated in 4.2
+ avatar: ActorImage
+
+ avatars: ActorImage[]
}
userHistory?: {
pluginData?: any
- static buildClientUrl (videoUUID: string) {
- return '/videos/watch/' + videoUUID
+ streamingPlaylists?: VideoStreamingPlaylist[]
+ files?: VideoFile[]
+
+ static buildWatchUrl (video: Partial<Pick<Video, 'uuid' | 'shortUUID'>>) {
+ return buildVideoWatchPath({ shortUUID: video.shortUUID || video.uuid })
+ }
+
+ static buildUpdateUrl (video: Pick<Video, 'uuid'>) {
+ return '/videos/update/' + video.uuid
}
- constructor (hash: VideoServerModel, translations = {}) {
+ constructor (hash: VideoServerModel, translations: { [ id: string ]: string } = {}) {
const absoluteAPIUrl = getAbsoluteAPIUrl()
this.createdAt = new Date(hash.createdAt.toString())
this.id = hash.id
this.uuid = hash.uuid
+ this.shortUUID = hash.shortUUID
this.isLocal = hash.isLocal
this.name = hash.name
this.url = hash.url
this.views = hash.views
+ this.viewers = hash.viewers
this.likes = hash.likes
this.dislikes = hash.dislikes
this.byAccount = Actor.CREATE_BY_STRING(hash.account.name, hash.account.host)
this.byVideoChannel = Actor.CREATE_BY_STRING(hash.channel.name, hash.channel.host)
- this.videoChannelAvatarUrl = VideoChannel.GET_ACTOR_AVATAR_URL(this.channel)
this.category.label = peertubeTranslate(this.category.label, translations)
this.licence.label = peertubeTranslate(this.licence.label, translations)
if (this.state) this.state.label = peertubeTranslate(this.state.label, translations)
this.blacklisted = hash.blacklisted
- this.blockedReason = hash.blacklistedReason
+ this.blacklistedReason = hash.blacklistedReason
+
+ this.blockedOwner = hash.blockedOwner
+ this.blockedServer = hash.blockedServer
+
+ this.streamingPlaylists = hash.streamingPlaylists
+ this.files = hash.files
this.userHistory = hash.userHistory
this.pluginData = hash.pluginData
}
- isVideoNSFWForUser (user: User, serverConfig: ServerConfig) {
+ isVideoNSFWForUser (user: User, serverConfig: HTMLServerConfig) {
// Video is not NSFW, skip
if (this.nsfw === false) return false
return user && this.isLocal === true && (this.account.name === user.username || user.hasRight(UserRight.UPDATE_ANY_VIDEO))
}
+ isEditableBy (user: AuthUser, videoStudioEnabled: boolean) {
+ return videoStudioEnabled &&
+ this.state?.id === VideoState.PUBLISHED &&
+ this.isUpdatableBy(user)
+ }
+
+ canSeeStats (user: AuthUser) {
+ return user && this.isLocal === true && (this.account.name === user.username || user.hasRight(UserRight.SEE_ALL_VIDEOS))
+ }
+
+ canRemoveFiles (user: AuthUser) {
+ return this.isLocal &&
+ user && user.hasRight(UserRight.MANAGE_VIDEO_FILES) &&
+ this.state.id !== VideoState.TO_TRANSCODE &&
+ this.hasHLS() &&
+ this.hasWebTorrent()
+ }
+
+ canRunTranscoding (user: AuthUser) {
+ return this.isLocal &&
+ user && user.hasRight(UserRight.RUN_VIDEO_TRANSCODING) &&
+ this.state.id !== VideoState.TO_TRANSCODE
+ }
+
+ hasHLS () {
+ return this.streamingPlaylists?.some(p => p.type === VideoStreamingPlaylistType.HLS)
+ }
+
+ hasWebTorrent () {
+ return this.files && this.files.length !== 0
+ }
+
isLiveInfoAvailableBy (user: AuthUser) {
return this.isLive &&
user && this.isLocal === true && (this.account.name === user.username || user.hasRight(UserRight.GET_ANY_LIVE))
}
getExactNumberOfViews () {
- if (this.views < 1000) return ''
-
if (this.isLive) {
- return $localize`${this.views} viewers`
+ return Video.viewersICU({ viewers: this.viewers }, $localize`${this.viewers} viewer(s)`)
}
- return $localize`${this.views} views`
+ return Video.viewsICU({ views: this.views }, $localize`{${this.views} view(s)}`)
}
}