import { AuthUser } from '@app/core'
import { User } from '@app/core/users/user.model'
import { durationToString, getAbsoluteAPIUrl, getAbsoluteEmbedUrl } from '@app/helpers'
+import { Actor } from '@app/shared/shared-main/account/actor.model'
+import { buildVideoWatchPath } from '@shared/core-utils'
import { peertubeTranslate } from '@shared/core-utils/i18n'
import {
- Avatar,
- ServerConfig,
+ ActorImage,
+ HTMLServerConfig,
UserRight,
Video as VideoServerModel,
VideoConstant,
+ VideoFile,
VideoPrivacy,
VideoScheduleUpdate,
- VideoState
+ VideoState,
+ VideoStreamingPlaylist,
+ VideoStreamingPlaylistType
} from '@shared/models'
-import { Actor } from '../account/actor.model'
export class Video implements VideoServerModel {
byVideoChannel: string
byAccount: string
- accountAvatarUrl: 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
thumbnailUrl: string
+ isLive: boolean
+
previewPath: string
previewUrl: string
embedPath: string
embedUrl: string
- url?: string
+ url: string
views: number
+ // If live
+ 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?: Avatar
+ avatar?: ActorImage
}
channel: {
displayName: string
url: string
host: string
- avatar?: Avatar
+ avatar?: ActorImage
}
userHistory?: {
currentTime: number
}
- static buildClientUrl (videoUUID: string) {
- return '/videos/watch/' + videoUUID
+ pluginData?: any
+
+ 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.state = hash.state
this.description = hash.description
+ this.isLive = hash.isLive
+
this.duration = hash.duration
this.durationLabel = durationToString(hash.duration)
this.id = hash.id
this.uuid = hash.uuid
+ this.shortUUID = hash.shortUUID
this.isLocal = hash.isLocal
this.name = hash.name
this.thumbnailPath = hash.thumbnailPath
- this.thumbnailUrl = hash.thumbnailUrl || (absoluteAPIUrl + hash.thumbnailPath)
+ this.thumbnailUrl = this.thumbnailPath
+ ? hash.thumbnailUrl || (absoluteAPIUrl + hash.thumbnailPath)
+ : null
this.previewPath = hash.previewPath
- this.previewUrl = hash.previewUrl || (absoluteAPIUrl + hash.previewPath)
+ this.previewUrl = this.previewPath
+ ? hash.previewUrl || (absoluteAPIUrl + hash.previewPath)
+ : null
this.embedPath = hash.embedPath
this.embedUrl = hash.embedUrl || (getAbsoluteEmbedUrl() + hash.embedPath)
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.accountAvatarUrl = Actor.GET_ACTOR_AVATAR_URL(this.account)
- this.videoChannelAvatarUrl = Actor.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.originInstanceHost = this.account.host
this.originInstanceUrl = 'https://' + this.originInstanceHost
+
+ 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))
}
+ canRemoveFiles (user: AuthUser) {
+ return this.isLocal &&
+ user.hasRight(UserRight.MANAGE_VIDEO_FILES) &&
+ this.state.id !== VideoState.TO_TRANSCODE &&
+ this.hasHLS() &&
+ this.hasWebTorrent()
+ }
+
+ canRunTranscoding (user: AuthUser) {
+ return this.isLocal &&
+ 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))
+ }
+
canBeDuplicatedBy (user: AuthUser) {
return user && this.isLocal === false && user.hasRight(UserRight.MANAGE_VIDEOS_REDUNDANCIES)
}
+
+ getExactNumberOfViews () {
+ if (this.views < 1000) return ''
+
+ if (this.isLive) {
+ return $localize`${this.views} viewers`
+ }
+
+ return $localize`${this.views} views`
+ }
}