* Implement ads in desktop app * Finish ads * use git dep instead * attempt to fix linux build (temp) * bump version + lint * comment more * fix build * try to fix linux build * Fix crashing on windows * Fix icons not showing * Remove useless env vars * Actual linux build fix * Run fmt * Fix scrolling * fix clippy * bump version + fix localhost * rev linux build patch * update version num * update csp * update csp * update csp * Switch to mousewheel event
145 lines
3.5 KiB
Vue
145 lines
3.5 KiB
Vue
<script setup>
|
|
import { ref, onMounted, onUnmounted, computed } from 'vue'
|
|
import { useRoute } from 'vue-router'
|
|
import RowDisplay from '@/components/RowDisplay.vue'
|
|
import { list } from '@/helpers/profile.js'
|
|
import { profile_listener } from '@/helpers/events'
|
|
import { useBreadcrumbs } from '@/store/breadcrumbs'
|
|
import { handleError } from '@/store/notifications.js'
|
|
import dayjs from 'dayjs'
|
|
import { get_search_results } from '@/helpers/cache.js'
|
|
import { hide_ads_window } from '@/helpers/ads.js'
|
|
|
|
onMounted(() => {
|
|
hide_ads_window()
|
|
})
|
|
|
|
const featuredModpacks = ref({})
|
|
const featuredMods = ref({})
|
|
const filter = ref('')
|
|
|
|
const route = useRoute()
|
|
const breadcrumbs = useBreadcrumbs()
|
|
|
|
breadcrumbs.setRootContext({ name: 'Home', link: route.path })
|
|
|
|
const recentInstances = ref([])
|
|
|
|
const offline = ref(!navigator.onLine)
|
|
window.addEventListener('offline', () => {
|
|
offline.value = true
|
|
})
|
|
window.addEventListener('online', () => {
|
|
offline.value = false
|
|
})
|
|
|
|
const getInstances = async () => {
|
|
const profiles = await list().catch(handleError)
|
|
|
|
recentInstances.value = profiles.sort((a, b) => {
|
|
const dateA = dayjs(a.last_played ?? 0)
|
|
const dateB = dayjs(b.last_played ?? 0)
|
|
|
|
if (dateA.isSame(dateB)) {
|
|
return a.name.localeCompare(b.name)
|
|
}
|
|
|
|
return dateB - dateA
|
|
})
|
|
|
|
let filters = []
|
|
for (const instance of recentInstances.value) {
|
|
if (instance.linked_data && instance.linked_data.project_id) {
|
|
filters.push(`NOT"project_id"="${instance.linked_data.project_id}"`)
|
|
}
|
|
}
|
|
filter.value = filters.join(' AND ')
|
|
}
|
|
|
|
const getFeaturedModpacks = async () => {
|
|
const response = await get_search_results(
|
|
`?facets=[["project_type:modpack"]]&limit=10&index=follows&filters=${filter.value}`,
|
|
)
|
|
|
|
if (response) {
|
|
featuredModpacks.value = response.result.hits
|
|
} else {
|
|
featuredModpacks.value = []
|
|
}
|
|
}
|
|
const getFeaturedMods = async () => {
|
|
const response = await get_search_results('?facets=[["project_type:mod"]]&limit=10&index=follows')
|
|
|
|
if (response) {
|
|
featuredMods.value = response.result.hits
|
|
} else {
|
|
featuredModpacks.value = []
|
|
}
|
|
}
|
|
|
|
await getInstances()
|
|
|
|
await Promise.all([getFeaturedModpacks(), getFeaturedMods()])
|
|
|
|
const unlistenProfile = await profile_listener(async (e) => {
|
|
await getInstances()
|
|
|
|
if (e.event === 'added' || e.event === 'created' || e.event === 'removed') {
|
|
await Promise.all([getFeaturedModpacks(), getFeaturedMods()])
|
|
}
|
|
})
|
|
|
|
// computed sums of recentInstances, featuredModpacks, featuredMods, treating them as arrays if they are not
|
|
const total = computed(() => {
|
|
return (
|
|
(recentInstances.value?.length ?? 0) +
|
|
(featuredModpacks.value?.length ?? 0) +
|
|
(featuredMods.value?.length ?? 0)
|
|
)
|
|
})
|
|
|
|
onUnmounted(() => {
|
|
unlistenProfile()
|
|
})
|
|
</script>
|
|
|
|
<template>
|
|
<div class="page-container">
|
|
<RowDisplay
|
|
v-if="total > 0"
|
|
:instances="[
|
|
{
|
|
label: 'Jump back in',
|
|
route: '/library',
|
|
instances: recentInstances,
|
|
instance: true,
|
|
downloaded: true,
|
|
},
|
|
{
|
|
label: 'Popular packs',
|
|
route: '/browse/modpack',
|
|
instances: featuredModpacks,
|
|
downloaded: false,
|
|
},
|
|
{
|
|
label: 'Popular mods',
|
|
route: '/browse/mod',
|
|
instances: featuredMods,
|
|
downloaded: false,
|
|
},
|
|
]"
|
|
:can-paginate="true"
|
|
/>
|
|
</div>
|
|
</template>
|
|
|
|
<style lang="scss" scoped>
|
|
.page-container {
|
|
display: flex;
|
|
flex-direction: column;
|
|
align-items: center;
|
|
justify-content: center;
|
|
width: 100%;
|
|
}
|
|
</style>
|