Piped/src/components/VideoItem.vue

210 lines
7.7 KiB
Vue
Raw Normal View History

2021-06-16 19:14:46 +00:00
<template>
<div class="flex flex-col flex-justify-between" v-if="showVideo">
2022-04-08 21:29:50 +00:00
<router-link
2023-03-19 10:10:21 +00:00
class="focus:underline hover:underline inline-block w-full"
2022-04-08 21:29:50 +00:00
:to="{
path: '/watch',
query: {
2022-11-01 12:12:54 +00:00
v: item.url.substr(-11),
2022-04-08 21:29:50 +00:00
...(playlistId && { list: playlistId }),
...(index >= 0 && { index: index + 1 }),
},
}"
>
2023-02-16 17:26:14 +00:00
<div class="w-full">
<img
2023-03-04 08:49:53 +00:00
class="w-full aspect-video object-contain"
2023-07-19 00:40:12 +00:00
:src="thumbnail"
:alt="title"
2023-03-01 16:06:10 +00:00
:class="{ 'shorts-img': item.isShort, 'opacity-75': item.watched }"
2023-02-16 17:26:14 +00:00
loading="lazy"
/>
<!-- progress bar -->
<div class="relative w-full h-1">
<div
class="absolute bottom-0 left-0 h-1 bg-red-600"
v-if="item.watched && item.duration > 0"
:style="{ width: `clamp(0%, ${(item.currentTime / item.duration) * 100}%, 100%` }"
/>
</div>
</div>
2021-12-27 14:46:33 +00:00
<div class="relative text-sm">
2021-12-27 16:29:25 +00:00
<span
class="thumbnail-overlay thumbnail-right"
2022-11-01 12:12:54 +00:00
v-if="item.duration > 0"
v-text="timeFormat(item.duration)"
2021-12-27 16:29:25 +00:00
/>
<!-- shorts thumbnail -->
2022-11-01 12:12:54 +00:00
<span class="thumbnail-overlay thumbnail-left" v-if="item.isShort" v-t="'video.shorts'" />
<span
class="thumbnail-overlay thumbnail-right"
v-else-if="item.duration >= 0"
2022-11-01 12:12:54 +00:00
v-text="timeFormat(item.duration)"
/>
<i18n-t v-else keypath="video.live" class="thumbnail-overlay thumbnail-right !bg-red-600" tag="div">
2022-08-17 13:34:57 +00:00
<font-awesome-icon class="w-3" :icon="['fas', 'broadcast-tower']" />
</i18n-t>
2022-11-01 12:12:54 +00:00
<span v-if="item.watched" class="thumbnail-overlay bottom-5px left-5px" v-t="'video.watched'" />
</div>
<div>
2021-10-07 22:44:54 +00:00
<p
2021-12-28 14:39:20 +00:00
style="display: -webkit-box; -webkit-line-clamp: 2; -webkit-box-orient: vertical"
2023-03-19 10:10:21 +00:00
class="pt-2 overflow-hidden flex link font-bold"
2023-07-19 00:40:12 +00:00
:title="title"
v-text="title"
2021-12-27 16:29:25 +00:00
/>
</div>
2021-06-16 19:14:46 +00:00
</router-link>
2022-01-12 11:59:50 +00:00
<div class="flex">
2022-11-01 12:12:54 +00:00
<router-link :to="item.uploaderUrl">
2021-10-07 22:44:54 +00:00
<img
2022-11-01 12:12:54 +00:00
v-if="item.uploaderAvatar"
:src="item.uploaderAvatar"
2021-10-07 22:44:54 +00:00
loading="lazy"
2021-12-27 14:46:29 +00:00
class="rounded-full mr-0.5 mt-0.5 w-32px h-32px"
2021-12-27 14:46:31 +00:00
width="68"
height="68"
2021-10-07 22:44:54 +00:00
/>
</router-link>
2023-03-19 10:10:21 +00:00
<div class="px-2 flex-1">
2021-10-07 22:44:54 +00:00
<router-link
2022-11-01 12:12:54 +00:00
v-if="item.uploaderUrl && item.uploaderName && !hideChannel"
2023-03-19 10:10:21 +00:00
class="link-secondary overflow-hidden block text-sm"
2022-11-01 12:12:54 +00:00
:to="item.uploaderUrl"
:title="item.uploaderName"
2021-10-07 22:44:54 +00:00
>
2022-11-01 12:12:54 +00:00
<span v-text="item.uploaderName" />
<font-awesome-icon class="ml-1.5" v-if="item.uploaderVerified" icon="check" />
2021-06-16 19:14:46 +00:00
</router-link>
2023-03-19 10:10:21 +00:00
<div v-if="item.views >= 0 || item.uploadedDate" class="text-xs font-normal text-gray-300 mt-1">
2022-11-01 12:12:54 +00:00
<span v-if="item.views >= 0">
2021-12-27 22:33:55 +00:00
<font-awesome-icon icon="eye" />
2023-03-19 10:10:21 +00:00
<span class="pl-1" v-text="`${numberFormat(item.views)} `" />
</span>
2022-11-01 12:12:54 +00:00
<span v-if="item.uploaded > 0" class="pl-0.5" v-text="timeAgo(item.uploaded)" />
<span v-else-if="item.uploadedDate" class="pl-0.5" v-text="item.uploadedDate" />
2023-03-19 10:10:21 +00:00
</div>
</div>
2023-03-19 14:33:07 +00:00
<div class="flex items-center gap-2.5">
2023-03-19 10:10:21 +00:00
<router-link
:to="{
path: '/watch',
query: {
v: item.url.substr(-11),
...(playlistId && { list: playlistId }),
...(index >= 0 && { index: index + 1 }),
listen: '1',
},
}"
2023-07-19 00:40:12 +00:00
:aria-label="'Listen to ' + title"
:title="'Listen to ' + title"
2023-03-19 10:10:21 +00:00
>
<font-awesome-icon icon="headphones" />
</router-link>
2023-06-15 17:18:47 +00:00
<button :title="$t('actions.add_to_playlist')" @click="showModal = !showModal">
2023-03-19 10:10:21 +00:00
<font-awesome-icon icon="circle-plus" />
</button>
<button
v-if="admin"
:title="$t('actions.remove_from_playlist')"
ref="removeButton"
@click="showConfirmRemove = true"
2023-03-19 10:10:21 +00:00
>
<font-awesome-icon icon="circle-minus" />
</button>
<ConfirmModal
v-if="showConfirmRemove"
@close="showConfirmRemove = false"
@confirm="removeVideo(item.url.substr(-11))"
:message="$t('actions.delete_playlist_video_confirm')"
/>
<PlaylistAddModal
v-if="showModal"
:video-id="item.url.substr(-11)"
2023-06-15 17:18:47 +00:00
:video-info="item"
@close="showModal = !showModal"
/>
2021-06-16 19:14:46 +00:00
</div>
</div>
</div>
</template>
2021-12-27 14:46:29 +00:00
<style>
.shorts-img {
2023-03-19 14:36:25 +00:00
@apply w-full object-contain;
}
2021-12-27 14:46:29 +00:00
</style>
2021-06-16 19:14:46 +00:00
<script>
2022-04-07 02:33:25 +00:00
import PlaylistAddModal from "./PlaylistAddModal.vue";
import ConfirmModal from "./ConfirmModal.vue";
2022-04-07 02:33:25 +00:00
2021-06-16 19:14:46 +00:00
export default {
props: {
2022-11-01 12:12:54 +00:00
item: {
type: Object,
default: () => {
return {};
},
},
2022-09-10 17:08:50 +00:00
isFeed: {
type: Boolean,
default: false,
},
height: { type: String, default: "118" },
width: { type: String, default: "210" },
hideChannel: { type: Boolean, default: false },
2022-04-07 02:33:25 +00:00
index: { type: Number, default: -1 },
playlistId: { type: String, default: null },
admin: { type: Boolean, default: false },
},
data() {
return {
showModal: false,
2022-09-10 17:08:50 +00:00
showVideo: true,
showConfirmRemove: false,
2022-04-07 02:33:25 +00:00
};
},
2022-09-10 17:08:50 +00:00
mounted() {
this.shouldShowVideo();
},
2023-07-19 00:40:12 +00:00
computed: {
title() {
return this.item.dearrow?.titles[0]?.title ?? this.item.title;
},
thumbnail() {
return this.item.dearrow?.thumbnails[0]?.thumbnail ?? this.item.thumbnail;
},
},
2022-04-07 02:33:25 +00:00
methods: {
removeVideo() {
this.$refs.removeButton.disabled = true;
this.removeVideoFromPlaylist(this.playlistId, this.index).then(json => {
if (json.error) alert(json.error);
else this.$emit("remove");
});
2022-04-07 02:33:25 +00:00
},
2022-09-10 17:08:50 +00:00
shouldShowVideo() {
if (!this.isFeed || !this.getPreferenceBoolean("hideWatched", false)) return;
const objectStore = window.db.transaction("watch_history", "readonly").objectStore("watch_history");
2022-11-01 12:12:54 +00:00
const request = objectStore.get(this.item.url.substr(-11));
request.onsuccess = event => {
const video = event.target.result;
if (video && (video.currentTime ?? 0) > video.duration * 0.9) {
this.showVideo = false;
return;
2022-09-10 17:08:50 +00:00
}
};
},
2021-06-16 19:14:46 +00:00
},
components: { PlaylistAddModal, ConfirmModal },
2021-06-16 19:14:46 +00:00
};
</script>