mirror of
https://github.com/TeamPiped/Piped.git
synced 2024-08-14 23:57:27 +00:00
Merge pull request #1652 from TeamPiped/content-item
Implement content item component
This commit is contained in:
commit
88d04a0bdd
12 changed files with 142 additions and 91 deletions
34
src/components/ChannelItem.vue
Normal file
34
src/components/ChannelItem.vue
Normal 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>
|
|
@ -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
|
||||||
|
|
|
@ -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>
|
||||||
|
|
|
@ -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>
|
||||||
|
|
||||||
|
|
|
@ -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 />
|
||||||
|
|
34
src/components/PlaylistItem.vue
Normal file
34
src/components/PlaylistItem.vue
Normal 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>
|
|
@ -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"
|
||||||
|
|
|
@ -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"
|
||||||
|
|
|
@ -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>
|
||||||
|
|
|
@ -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>
|
||||||
|
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -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,
|
||||||
|
|
Loading…
Reference in a new issue