Merge pull request #1652 from TeamPiped/content-item

Implement content item component
This commit is contained in:
Kavin 2022-11-01 13:30:12 +00:00 committed by GitHub
commit 88d04a0bdd
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
12 changed files with 142 additions and 91 deletions

View file

@ -0,0 +1,34 @@
<template>
<div>
<router-link :to="props.item.url">
<div class="relative">
<img class="w-full" :src="props.item.thumbnail" loading="lazy" />
</div>
<p>
<span v-text="props.item.name" />
<font-awesome-icon class="ml-1.5" v-if="props.item.verified" icon="check" />
</p>
</router-link>
<p v-if="props.item.description" v-text="props.item.description" />
<router-link v-if="props.item.uploaderUrl" class="link" :to="props.item.uploaderUrl">
<p>
<span v-text="props.item.uploader" />
<font-awesome-icon class="ml-1.5" v-if="props.item.uploaderVerified" icon="check" />
</p>
</router-link>
<a v-if="props.item.uploaderName" class="link" v-text="props.item.uploaderName" />
<template v-if="props.item.videos >= 0">
<br v-if="props.item.uploaderName" />
<strong v-text="`${props.item.videos} ${$t('video.videos')}`" />
</template>
<br />
</div>
</template>
<script setup>
const props = defineProps({
item: Object,
});
</script>

View file

@ -53,7 +53,7 @@
<ContentItem <ContentItem
v-for="item in contentItems" v-for="item in contentItems"
:key="item.url" :key="item.url"
:content-item="item" :item="item"
height="94" height="94"
width="168" width="168"
hide-channel hide-channel

View file

@ -1,52 +1,35 @@
<template> <template>
<VideoItem v-if="shouldUseVideoItem(contentItem)" :video="contentItem" height="94" width="168" /> <component :is="compName" :item="item" />
<div v-else>
<router-link :to="contentItem.url">
<div class="relative">
<img class="w-full" :src="contentItem.thumbnail" loading="lazy" />
</div>
<p>
<span v-text="contentItem.name" />
<font-awesome-icon class="ml-1.5" v-if="contentItem.verified" icon="check" />
</p>
</router-link>
<p v-if="contentItem.description" v-text="contentItem.description" />
<router-link v-if="contentItem.uploaderUrl" class="link" :to="contentItem.uploaderUrl">
<p>
<span v-text="contentItem.uploader" />
<font-awesome-icon class="ml-1.5" v-if="contentItem.uploaderVerified" icon="check" />
</p>
</router-link>
<a v-if="contentItem.uploaderName" class="link" v-text="contentItem.uploaderName" />
<template v-if="contentItem.videos >= 0">
<br v-if="contentItem.uploaderName" />
<strong v-text="`${contentItem.videos} ${$t('video.videos')}`" />
</template> </template>
<br /> <script setup>
</div> import { defineAsyncComponent } from "vue";
</template>
<script> const props = defineProps({
import VideoItem from "./VideoItem.vue"; item: Object,
});
export default { const VideoItem = defineAsyncComponent(() => import("./VideoItem.vue"));
components: { const PlaylistItem = defineAsyncComponent(() => import("./PlaylistItem.vue"));
VideoItem, const ChannelItem = defineAsyncComponent(() => import("./ChannelItem.vue"));
},
props: { var compName;
contentItem: {
type: Object, switch (props.item?.type) {
default: () => { case "stream":
return {}; compName = VideoItem;
}, break;
}, case "playlist":
}, compName = PlaylistItem;
methods: { break;
shouldUseVideoItem(item) { case "channel":
return item.title; compName = ChannelItem;
}, break;
}, default:
}; console.error("Unexpected item type: " + props.item.type);
}
defineExpose({
compName,
});
</script> </script>

View file

@ -18,7 +18,7 @@
<hr /> <hr />
<div class="video-grid"> <div class="video-grid">
<VideoItem :is-feed="true" v-for="video in videos" :key="video.url" :video="video" /> <VideoItem :is-feed="true" v-for="video in videos" :key="video.url" :item="video" />
</div> </div>
</template> </template>

View file

@ -14,7 +14,7 @@
<hr /> <hr />
<div class="video-grid"> <div class="video-grid">
<VideoItem v-for="video in videos" :key="video.url" :video="video" /> <VideoItem v-for="video in videos" :key="video.url" :item="video" />
</div> </div>
<br /> <br />

View file

@ -0,0 +1,34 @@
<template>
<div>
<router-link :to="props.item.url">
<div class="relative">
<img class="w-full" :src="props.item.thumbnail" loading="lazy" />
</div>
<p>
<span v-text="props.item.name" />
<font-awesome-icon class="ml-1.5" v-if="props.item.verified" icon="check" />
</p>
</router-link>
<p v-if="props.item.description" v-text="props.item.description" />
<router-link v-if="props.item.uploaderUrl" class="link" :to="props.item.uploaderUrl">
<p>
<span v-text="props.item.uploader" />
<font-awesome-icon class="ml-1.5" v-if="props.item.uploaderVerified" icon="check" />
</p>
</router-link>
<a v-if="props.item.uploaderName" class="link" v-text="props.item.uploaderName" />
<template v-if="props.item.videos >= 0">
<br v-if="props.item.uploaderName" />
<strong v-text="`${props.item.videos} ${$t('video.videos')}`" />
</template>
<br />
</div>
</template>
<script setup>
const props = defineProps({
item: Object,
});
</script>

View file

@ -32,7 +32,7 @@
<VideoItem <VideoItem
v-for="(video, index) in playlist.relatedStreams" v-for="(video, index) in playlist.relatedStreams"
:key="video.url" :key="video.url"
:video="video" :item="video"
:index="index" :index="index"
:playlist-id="$route.query.list" :playlist-id="$route.query.list"
:admin="admin" :admin="admin"

View file

@ -3,7 +3,7 @@
<VideoItem <VideoItem
v-for="(related, index) in playlist.relatedStreams" v-for="(related, index) in playlist.relatedStreams"
:key="related.url" :key="related.url"
:video="related" :item="related"
:index="index" :index="index"
:playlist-id="playlistId" :playlist-id="playlistId"
height="94" height="94"

View file

@ -20,7 +20,7 @@
<div v-if="results" class="video-grid"> <div v-if="results" class="video-grid">
<template v-for="result in results.items" :key="result.url"> <template v-for="result in results.items" :key="result.url">
<ContentItem :content-item="result" /> <ContentItem :item="result" height="94" width="168" />
</template> </template>
</div> </div>
</template> </template>

View file

@ -4,7 +4,7 @@
<hr /> <hr />
<div class="video-grid"> <div class="video-grid">
<VideoItem v-for="video in videos" :key="video.url" :video="video" height="118" width="210" /> <VideoItem v-for="video in videos" :key="video.url" :item="video" height="118" width="210" />
</div> </div>
</template> </template>

View file

@ -4,7 +4,7 @@
:to="{ :to="{
path: '/watch', path: '/watch',
query: { query: {
v: video.url.substr(-11), v: item.url.substr(-11),
...(playlistId && { list: playlistId }), ...(playlistId && { list: playlistId }),
...(index >= 0 && { index: index + 1 }), ...(index >= 0 && { index: index + 1 }),
}, },
@ -12,36 +12,36 @@
> >
<img <img
class="w-full" class="w-full"
:src="video.thumbnail" :src="item.thumbnail"
:alt="video.title" :alt="item.title"
:class="{ 'shorts-img': video.isShort }" :class="{ 'shorts-img': item.isShort }"
loading="lazy" loading="lazy"
/> />
<div class="relative text-sm"> <div class="relative text-sm">
<span <span
class="thumbnail-overlay thumbnail-right" class="thumbnail-overlay thumbnail-right"
v-if="video.duration > 0" v-if="item.duration > 0"
v-text="timeFormat(video.duration)" v-text="timeFormat(item.duration)"
/> />
<!-- shorts thumbnail --> <!-- shorts thumbnail -->
<span class="thumbnail-overlay thumbnail-left" v-if="video.isShort" v-t="'video.shorts'" /> <span class="thumbnail-overlay thumbnail-left" v-if="item.isShort" v-t="'video.shorts'" />
<span <span
class="thumbnail-overlay thumbnail-right" class="thumbnail-overlay thumbnail-right"
v-else-if="video.duration >= 60" v-else-if="item.duration >= 60"
v-text="timeFormat(video.duration)" v-text="timeFormat(item.duration)"
/> />
<i18n-t v-else keypath="video.live" class="thumbnail-overlay thumbnail-right !bg-red-600" tag="div"> <i18n-t v-else keypath="video.live" class="thumbnail-overlay thumbnail-right !bg-red-600" tag="div">
<font-awesome-icon class="w-3" :icon="['fas', 'broadcast-tower']" /> <font-awesome-icon class="w-3" :icon="['fas', 'broadcast-tower']" />
</i18n-t> </i18n-t>
<span v-if="video.watched" class="thumbnail-overlay bottom-5px left-5px" v-t="'video.watched'" /> <span v-if="item.watched" class="thumbnail-overlay bottom-5px left-5px" v-t="'video.watched'" />
</div> </div>
<div> <div>
<p <p
style="display: -webkit-box; -webkit-line-clamp: 2; -webkit-box-orient: vertical" style="display: -webkit-box; -webkit-line-clamp: 2; -webkit-box-orient: vertical"
class="my-2 overflow-hidden flex link" class="my-2 overflow-hidden flex link"
:title="video.title" :title="item.title"
v-text="video.title" v-text="item.title"
/> />
</div> </div>
</router-link> </router-link>
@ -51,14 +51,14 @@
:to="{ :to="{
path: '/watch', path: '/watch',
query: { query: {
v: video.url.substr(-11), v: item.url.substr(-11),
...(playlistId && { list: playlistId }), ...(playlistId && { list: playlistId }),
...(index >= 0 && { index: index + 1 }), ...(index >= 0 && { index: index + 1 }),
listen: '1', listen: '1',
}, },
}" }"
:aria-label="'Listen to ' + video.title" :aria-label="'Listen to ' + item.title"
:title="'Listen to ' + video.title" :title="'Listen to ' + item.title"
> >
<font-awesome-icon icon="headphones" /> <font-awesome-icon icon="headphones" />
</router-link> </router-link>
@ -69,20 +69,20 @@
v-if="admin" v-if="admin"
:title="$t('actions.remove_from_playlist')" :title="$t('actions.remove_from_playlist')"
ref="removeButton" ref="removeButton"
@click="removeVideo(video.url.substr(-11))" @click="removeVideo(item.url.substr(-11))"
> >
<font-awesome-icon icon="circle-minus" /> <font-awesome-icon icon="circle-minus" />
</button> </button>
<PlaylistAddModal v-if="showModal" :video-id="video.url.substr(-11)" @close="showModal = !showModal" /> <PlaylistAddModal v-if="showModal" :video-id="item.url.substr(-11)" @close="showModal = !showModal" />
</div> </div>
<div class="flex"> <div class="flex">
<router-link :to="video.uploaderUrl"> <router-link :to="item.uploaderUrl">
<img <img
v-if="video.uploaderAvatar" v-if="item.uploaderAvatar"
:src="video.uploaderAvatar" :src="item.uploaderAvatar"
loading="lazy" loading="lazy"
:alt="video.uploaderName" :alt="item.uploaderName"
class="rounded-full mr-0.5 mt-0.5 w-32px h-32px" class="rounded-full mr-0.5 mt-0.5 w-32px h-32px"
width="68" width="68"
height="68" height="68"
@ -91,22 +91,22 @@
<div class="w-[calc(100%-32px-1rem)]"> <div class="w-[calc(100%-32px-1rem)]">
<router-link <router-link
v-if="video.uploaderUrl && video.uploaderName && !hideChannel" v-if="item.uploaderUrl && item.uploaderName && !hideChannel"
class="link-secondary overflow-hidden block" class="link-secondary overflow-hidden block"
:to="video.uploaderUrl" :to="item.uploaderUrl"
:title="video.uploaderName" :title="item.uploaderName"
> >
<span v-text="video.uploaderName" /> <span v-text="item.uploaderName" />
<font-awesome-icon class="ml-1.5" v-if="video.uploaderVerified" icon="check" /> <font-awesome-icon class="ml-1.5" v-if="item.uploaderVerified" icon="check" />
</router-link> </router-link>
<strong v-if="video.views >= 0 || video.uploadedDate" class="text-sm"> <strong v-if="item.views >= 0 || item.uploadedDate" class="text-sm">
<span v-if="video.views >= 0"> <span v-if="item.views >= 0">
<font-awesome-icon icon="eye" /> <font-awesome-icon icon="eye" />
<span class="pl-0.5" v-text="`${numberFormat(video.views)} •`" /> <span class="pl-0.5" v-text="`${numberFormat(item.views)} •`" />
</span> </span>
<span v-if="video.uploaded > 0" class="pl-0.5" v-text="timeAgo(video.uploaded)" /> <span v-if="item.uploaded > 0" class="pl-0.5" v-text="timeAgo(item.uploaded)" />
<span v-else-if="video.uploadedDate" class="pl-0.5" v-text="video.uploadedDate" /> <span v-else-if="item.uploadedDate" class="pl-0.5" v-text="item.uploadedDate" />
</strong> </strong>
</div> </div>
</div> </div>
@ -124,7 +124,7 @@ import PlaylistAddModal from "./PlaylistAddModal.vue";
export default { export default {
props: { props: {
video: { item: {
type: Object, type: Object,
default: () => { default: () => {
return {}; return {};
@ -174,7 +174,7 @@ export default {
if (!this.isFeed || !this.getPreferenceBoolean("hideWatched", false)) return; if (!this.isFeed || !this.getPreferenceBoolean("hideWatched", false)) return;
const objectStore = window.db.transaction("watch_history", "readonly").objectStore("watch_history"); const objectStore = window.db.transaction("watch_history", "readonly").objectStore("watch_history");
const request = objectStore.get(this.video.url.substr(-11)); const request = objectStore.get(this.item.url.substr(-11));
request.onsuccess = event => { request.onsuccess = event => {
const video = event.target.result; const video = event.target.result;
if (video && (video.currentTime ?? 0) > video.duration * 0.9) { if (video && (video.currentTime ?? 0) > video.duration * 0.9) {

View file

@ -200,10 +200,10 @@
/> />
<hr v-show="showRecs" /> <hr v-show="showRecs" />
<div v-show="showRecs"> <div v-show="showRecs">
<VideoItem <ContentItem
v-for="related in video.relatedStreams" v-for="related in video.relatedStreams"
:key="related.url" :key="related.url"
:video="related" :item="related"
height="94" height="94"
width="168" width="168"
/> />
@ -216,7 +216,7 @@
<script> <script>
import VideoPlayer from "./VideoPlayer.vue"; import VideoPlayer from "./VideoPlayer.vue";
import VideoItem from "./VideoItem.vue"; import ContentItem from "./ContentItem.vue";
import ErrorHandler from "./ErrorHandler.vue"; import ErrorHandler from "./ErrorHandler.vue";
import CommentItem from "./CommentItem.vue"; import CommentItem from "./CommentItem.vue";
import ChaptersBar from "./ChaptersBar.vue"; import ChaptersBar from "./ChaptersBar.vue";
@ -228,7 +228,7 @@ export default {
name: "App", name: "App",
components: { components: {
VideoPlayer, VideoPlayer,
VideoItem, ContentItem,
ErrorHandler, ErrorHandler,
CommentItem, CommentItem,
ChaptersBar, ChaptersBar,