Hangar/frontend/pages/index.vue
2021-06-06 22:25:56 -07:00

153 lines
5.6 KiB
Vue

<template>
<div>
<v-row>
<v-col cols="12" sm="8" md="5" offset-md="3">
<v-row justify="center" align="center">
<h1>Hangar</h1>
<v-subheader v-text="$t('hangar.subtitle')" />
</v-row>
<v-row justify="center" align="center">
<v-col cols="12">
<v-text-field v-model="filters.search" :label="$t('hangar.projectSearch.query', [projects.pagination.count])" clearable />
</v-col>
</v-row>
<v-row justify="center" align="center">
<v-col cols="12">
<ProjectList :projects="projects" :list-options.sync="options" />
</v-col>
</v-row>
</v-col>
<v-col cols="12" sm="2" md="2">
<HangarSponsor :sponsor="sponsor" />
<v-select />
<v-checkbox :label="$t('hangar.projectSearch.relevanceSort')" />
<v-list dense>
<v-subheader>Categories</v-subheader>
<v-list-item-group v-model="filters.categories" multiple>
<v-list-item v-for="cat in $store.getters.visibleCategories" :key="cat.apiName" :value="cat.apiName">
<v-list-item-icon>
<v-icon v-text="cat.icon" />
</v-list-item-icon>
<v-list-item-content>
<v-list-item-title v-text="$t(`project.category.${cat.apiName}`)" />
</v-list-item-content>
</v-list-item>
</v-list-item-group>
</v-list>
<v-list dense>
<v-subheader>Platforms</v-subheader>
<v-list-item-group>
<v-list-item v-for="(platform, i) in platforms" :key="i" active-class="">
<v-list-item-icon>
<v-icon v-text="`$vuetify.icons.${platform.name.toLowerCase()}`" />
</v-list-item-icon>
<v-list-item-content>
<v-list-item-title v-text="platform.name" />
</v-list-item-content>
</v-list-item>
</v-list-item-group>
</v-list>
</v-col>
</v-row>
</div>
</template>
<script lang="ts">
import { Component } from 'nuxt-property-decorator';
import { PaginatedResult, Project, Sponsor } from 'hangar-api';
import { IPlatform } from 'hangar-internal';
import { Context } from '@nuxt/types';
import { DataOptions } from 'vuetify';
import { ProjectList } from '~/components/projects';
import HangarSponsor from '~/components/layouts/Sponsor.vue';
import { RootState } from '~/store';
import { HangarComponent } from '~/components/mixins';
import { Platform, ProjectCategory } from '~/types/enums';
@Component({
components: {
ProjectList,
HangarSponsor,
},
})
export default class Home extends HangarComponent {
// TODO implement filtering
projects!: PaginatedResult<Project>;
sponsor!: Sponsor;
filters = {
search: null as string | null,
platforms: [] as Platform[],
categories: [] as ProjectCategory[],
};
options = {
itemsPerPage: 25,
page: 1,
} as DataOptions;
head() {
const meta = this.$seo.head('Home', null, this.$route, null);
meta.script.push({
type: 'application/ld+json',
json: {
'@context': 'https://schema.org',
'@type': 'WebSite',
url: this.$seo.baseUrl(),
potentialAction: {
'@type': 'SearchAction',
target: this.$seo.baseUrl() + '/?q={search_term_string}', // todo fix once search actually works
'query-input': 'required name=search_term_string',
},
},
});
return meta;
}
mounted() {
this.$watch('options', this.reloadProjectList, { deep: true });
this.$watch('filters', this.reloadProjectList, { deep: true });
}
get platforms(): IPlatform[] {
return Array.from((this.$store.state as RootState).platforms.values());
}
reloadProjectList() {
this.$api
.request<PaginatedResult<Project>>('projects', false, 'get', this.requestOptions)
.then((result) => {
this.projects = result;
})
.catch(this.$util.handleRequestError);
}
get requestOptions() {
const requestOptions: { [key: string]: any } = {
limit: this.options.itemsPerPage,
offset: (this.options.page - 1) * this.options.itemsPerPage,
category: this.filters.categories,
};
if (this.filters.search != null && this.filters.search.length > 0) {
requestOptions.q = this.filters.search;
}
return requestOptions;
}
async asyncData({ $api, $util }: Context) {
const res = await Promise.all<Sponsor, PaginatedResult<Project>>([
$api.requestInternal<Sponsor>(`data/sponsor`, false),
$api.request<PaginatedResult<Project>>('projects', false, 'get', { limit: 25, offset: 0 }),
]).catch($util.handlePageRequestError);
if (typeof res === 'undefined') {
return;
}
return { sponsor: res[0], projects: res[1] };
}
}
</script>