PeerTube/client/src/app/+my-library/+my-video-channels/my-video-channels.component.ts

177 lines
5.1 KiB
TypeScript
Raw Normal View History

2020-06-23 14:10:17 +02:00
import { ChartData } from 'chart.js'
import { max, maxBy, min, minBy } from 'lodash-es'
import { Subject } from 'rxjs'
import { debounceTime, mergeMap } from 'rxjs/operators'
2020-06-23 14:10:17 +02:00
import { Component, OnInit } from '@angular/core'
import { AuthService, ConfirmService, Notifier, ScreenService, User } from '@app/core'
import { VideoChannel, VideoChannelService } from '@app/shared/shared-main'
2018-04-26 16:11:38 +02:00
@Component({
templateUrl: './my-video-channels.component.html',
styleUrls: [ './my-video-channels.component.scss' ]
2018-04-26 16:11:38 +02:00
})
export class MyVideoChannelsComponent implements OnInit {
2020-07-23 21:30:04 +02:00
totalItems: number
2018-04-26 16:11:38 +02:00
videoChannels: VideoChannel[] = []
videoChannelsChartData: ChartData[]
2020-03-23 10:14:05 +01:00
videoChannelsMinimumDailyViews = 0
videoChannelsMaximumDailyViews: number
2018-04-26 16:11:38 +02:00
2020-07-23 21:30:04 +02:00
channelsSearch: string
channelsSearchChanged = new Subject<string>()
chartOptions: any
2018-04-26 16:11:38 +02:00
private user: User
constructor (
private authService: AuthService,
private notifier: Notifier,
2018-04-26 16:11:38 +02:00
private confirmService: ConfirmService,
2018-06-04 16:21:17 +02:00
private videoChannelService: VideoChannelService,
private screenService: ScreenService
) {}
2018-04-26 16:11:38 +02:00
ngOnInit () {
this.user = this.authService.getUser()
this.loadVideoChannels()
2020-07-23 21:30:04 +02:00
this.channelsSearchChanged
.pipe(debounceTime(500))
.subscribe(() => {
this.loadVideoChannels()
})
2018-04-26 16:11:38 +02:00
}
2020-03-23 10:14:05 +01:00
get isInSmallView () {
return this.screenService.isInSmallView()
}
resetSearch () {
2020-07-23 21:30:04 +02:00
this.channelsSearch = ''
this.onChannelsSearchChanged()
}
onChannelsSearchChanged () {
this.channelsSearchChanged.next()
}
2018-04-26 16:11:38 +02:00
async deleteVideoChannel (videoChannel: VideoChannel) {
const res = await this.confirmService.confirmWithInput(
$localize`Do you really want to delete ${videoChannel.displayName}?
It will delete ${videoChannel.videosCount} videos uploaded in this channel, and you will not be able to create another
channel with the same name (${videoChannel.name})!`,
$localize`Please type the display name of the video channel (${videoChannel.displayName}) to confirm`,
2020-11-10 10:52:05 +01:00
videoChannel.displayName,
$localize`Delete`
2018-04-26 16:11:38 +02:00
)
if (res === false) return
this.videoChannelService.removeVideoChannel(videoChannel)
.subscribe(
() => {
2018-04-26 16:11:38 +02:00
this.loadVideoChannels()
this.notifier.success($localize`Video channel ${videoChannel.displayName} deleted.`)
2018-04-26 16:11:38 +02:00
},
error => this.notifier.error(error.message)
2018-04-26 16:11:38 +02:00
)
}
private loadVideoChannels () {
this.authService.userInformationLoaded
.pipe(mergeMap(() => this.videoChannelService.listAccountVideoChannels(this.user.account, null, true, this.channelsSearch)))
2020-03-23 10:14:05 +01:00
.subscribe(res => {
this.videoChannels = res.data
2020-07-23 21:30:04 +02:00
this.totalItems = res.total
// chart data
this.videoChannelsChartData = this.videoChannels.map(v => ({
2020-03-23 10:14:05 +01:00
labels: v.viewsPerDay.map(day => day.date.toLocaleDateString()),
datasets: [
{
label: $localize`Views for the day`,
data: v.viewsPerDay.map(day => day.views),
fill: false,
borderColor: '#c6c6c6'
2020-03-23 10:14:05 +01:00
}
]
} as ChartData))
// chart options that depend on chart data:
// we don't want to skew values and have min at 0, so we define what the floor/ceiling is here
this.videoChannelsMinimumDailyViews = min(
// compute local minimum daily views for each channel, by their "views" attribute
this.videoChannels.map(v => minBy(
v.viewsPerDay,
day => day.views
).views) // the object returned is a ViewPerDate, so we still need to get the views attribute
)
this.videoChannelsMaximumDailyViews = max(
// compute local maximum daily views for each channel, by their "views" attribute
this.videoChannels.map(v => maxBy(
v.viewsPerDay,
day => day.views
).views) // the object returned is a ViewPerDate, so we still need to get the views attribute
)
this.buildChartOptions()
2020-03-23 10:14:05 +01:00
})
2018-04-26 16:11:38 +02:00
}
private buildChartOptions () {
this.chartOptions = {
legend: {
display: false
},
scales: {
xAxes: [{
display: false
}],
yAxes: [{
display: false,
ticks: {
min: Math.max(0, this.videoChannelsMinimumDailyViews - (3 * this.videoChannelsMaximumDailyViews / 100)),
max: Math.max(1, this.videoChannelsMaximumDailyViews)
}
}]
},
layout: {
padding: {
left: 15,
right: 15,
top: 10,
bottom: 0
}
},
elements: {
point: {
radius: 0
}
},
tooltips: {
mode: 'index',
intersect: false,
custom: function (tooltip: any) {
if (!tooltip) return
// disable displaying the color box
tooltip.displayColors = false
},
callbacks: {
label: (tooltip: any, data: any) => `${tooltip.value} views`
}
},
hover: {
mode: 'index',
intersect: false
}
}
}
2018-04-26 16:11:38 +02:00
}