safetwitch/src/views/HomepageView.vue

156 lines
4.6 KiB
Vue
Raw Normal View History

2023-03-19 19:51:55 -04:00
<script lang="ts">
2023-06-19 23:18:10 -04:00
import { ref } from 'vue'
2023-03-31 07:50:46 -04:00
import StreamPreviewVue from '@/components/StreamPreview.vue'
import ErrorMessage from '@/components/ErrorMessage.vue'
import LoadingScreen from '@/components/LoadingScreen.vue'
import CategoryPreview from '@/components/CategoryPreview.vue'
2023-03-29 10:02:04 -04:00
2023-06-19 23:18:10 -04:00
import { getEndpoint } from '@/mixins'
import type { CategoryPreview as CategoryPreviewInterface } from '@/types'
2023-03-19 19:51:55 -04:00
export default {
2023-05-12 10:21:31 -04:00
inject: ['protocol'],
2023-03-19 19:51:55 -04:00
async setup() {
2023-06-19 23:18:10 -04:00
let data = ref<CategoryPreviewInterface[]>()
2023-07-20 13:57:01 -04:00
let status = ref<'ok' | 'error'>()
2023-03-29 10:02:04 -04:00
2023-03-19 19:51:55 -04:00
return {
2023-03-29 10:02:04 -04:00
data,
2023-06-19 23:18:10 -04:00
status,
2023-03-31 07:50:46 -04:00
filterTags: '',
following: ref([])
2023-03-19 19:51:55 -04:00
}
},
methods: {
2023-03-25 12:53:10 -04:00
filterSearches(toFilter: string) {
2023-04-10 00:19:39 -04:00
const categories = this.$refs.categoryItem
const wantedTags: string[] = toFilter.toLowerCase().split(',').filter((v) => v.toLowerCase())
2023-03-25 12:53:10 -04:00
for (let category of categories as any) {
2023-03-29 10:26:55 -04:00
let tagElements = category.getElementsByTagName('span')
2023-03-25 12:53:10 -04:00
let tags = []
for (let tag of tagElements) {
tags.push(tag.innerText.toLowerCase())
}
// Create sets from the arrays
const [set1, set2] = [new Set(wantedTags), new Set(tags)]
2023-03-29 10:26:55 -04:00
const common = [...set1].filter((x) => set2.has(x))
if (common.length === wantedTags.length) {
category.style.display = ''
} else if (wantedTags[0] === '') {
category.style.display = ''
2023-03-25 12:53:10 -04:00
} else {
2023-03-29 10:26:55 -04:00
category.style.display = 'none'
2023-03-25 12:53:10 -04:00
}
}
2023-03-29 10:02:04 -04:00
},
getNextCategory() {
window.onscroll = async () => {
2023-03-29 10:26:55 -04:00
let bottomOfWindow =
document.documentElement.scrollTop + window.innerHeight ===
document.documentElement.offsetHeight
2023-03-29 10:02:04 -04:00
if (bottomOfWindow && this.data) {
const cursor = this.data[this.data.length - 1].cursor
2023-03-29 10:26:55 -04:00
if (!cursor) return
const res = await fetch(
2023-07-20 13:57:01 -04:00
`${this.protocol}${
import.meta.env.SAFETWITCH_BACKEND_DOMAIN
}/api/discover/?cursor=${cursor}`
2023-03-29 10:26:55 -04:00
)
if (!res.ok) {
throw new Error('Failed to fetch data')
}
2023-04-11 14:27:25 -04:00
const resData = await res.json()
2023-03-29 10:02:04 -04:00
2023-04-11 14:27:25 -04:00
for (let category of resData.data) {
2023-03-29 10:02:04 -04:00
this.data.push(category)
}
2023-03-29 10:26:55 -04:00
}
2023-03-29 10:02:04 -04:00
}
2023-03-19 19:51:55 -04:00
}
2023-03-29 10:02:04 -04:00
},
async mounted() {
2023-03-29 10:02:04 -04:00
this.getNextCategory()
2023-03-31 07:50:46 -04:00
let following = localStorage.getItem('following')
if (following) {
this.following = JSON.parse(following)
} else {
this.following = []
}
2023-07-20 13:57:01 -04:00
await getEndpoint('api/discover')
.catch(() => {
this.status = 'error'
})
.then((data: CategoryPreviewInterface[]) => {
this.data = data
})
2023-03-31 07:50:46 -04:00
},
components: {
StreamPreviewVue,
ErrorMessage,
LoadingScreen,
CategoryPreview
2023-03-19 19:51:55 -04:00
}
}
</script>
<template>
2023-06-19 23:18:10 -04:00
<loading-screen v-if="!data && status != 'error'"></loading-screen>
<error-message v-else-if="status == 'error'"></error-message>
2023-06-19 23:18:10 -04:00
<div v-else-if="data" class="max-w-5xl mx-auto">
2023-03-31 07:50:46 -04:00
<div v-if="following.length > 0" class="p-2 text-white">
<h1 class="font-bold text-5xl">Following</h1>
<p class="text-xl">Streamers you follow</p>
<ul class="flex overflow-x-scroll flex-nowrap h-80 space-x-1">
2023-04-10 00:19:39 -04:00
<li
v-for="streamer in following"
:key="streamer"
class="inline-block hover:scale-105 transition-transform"
>
2023-03-31 07:50:46 -04:00
<stream-preview-vue :name="streamer"></stream-preview-vue>
</li>
</ul>
</div>
2023-03-19 19:51:55 -04:00
<div class="p-2">
2023-07-20 13:57:01 -04:00
<h1 class="font-bold text-5xl text-white">{{ $t('home.discover') }}</h1>
<p class="text-xl text-white">{{ $t('home.discoverDescription') }}</p>
2023-03-19 19:51:55 -04:00
<div class="pt-5 inline-flex text-white">
2023-07-20 13:57:01 -04:00
<p class="mr-2 font-bold text-white">{{ $t('home.tagDescription') }}</p>
<div class="relative">
2023-07-20 13:57:01 -04:00
<label for="searchBar" class="hidden">{{ $t('main.search') }}</label>
<v-icon name="io-search-outline" class="absolute my-auto inset-y-0 left-2"></v-icon>
2023-03-19 19:51:55 -04:00
<input
type="text"
id="searchBar"
name="searchBar"
2023-06-13 12:08:43 -04:00
:placeholder="$t('main.search')"
2023-03-25 12:53:10 -04:00
v-model="filterTags"
@keyup="filterSearches(filterTags)"
2023-03-19 19:51:55 -04:00
class="rounded-md p-1 pl-8 text-black bg-neutral-500 placeholder:text-white"
/>
</div>
2023-03-19 19:51:55 -04:00
</div>
</div>
2023-03-25 12:53:10 -04:00
<ul ref="categoryList">
<li
v-for="category in data"
2023-07-20 13:57:01 -04:00
:key="category.name"
2023-03-25 12:53:10 -04:00
ref="categoryItem"
class="inline-flex m-2 hover:scale-105 transition-transform"
>
2023-07-20 13:57:01 -04:00
<category-preview :category-data="category"></category-preview>
2023-03-19 19:51:55 -04:00
</li>
</ul>
</div>
</template>