2021-01-21 11:36:18 +08:00
|
|
|
<template>
|
2021-01-22 06:24:50 +08:00
|
|
|
<div>
|
|
|
|
<v-row>
|
2022-01-25 01:25:47 +08:00
|
|
|
<v-col cols="12" sm="8" md="5" offset-md="3" class="main-plugin-content">
|
2021-12-04 01:28:35 +08:00
|
|
|
<br />
|
|
|
|
<br />
|
2021-01-22 06:24:50 +08:00
|
|
|
<v-row justify="center" align="center">
|
|
|
|
<h1>Hangar</h1>
|
2021-12-08 15:44:29 +08:00
|
|
|
<v-subheader style="position: relative; top: 3px" v-text="$t('hangar.subtitle')" />
|
2021-01-22 06:24:50 +08:00
|
|
|
</v-row>
|
|
|
|
<v-row justify="center" align="center">
|
|
|
|
<v-col cols="12">
|
2021-06-07 13:25:56 +08:00
|
|
|
<v-text-field v-model="filters.search" :label="$t('hangar.projectSearch.query', [projects.pagination.count])" clearable />
|
2021-01-22 06:24:50 +08:00
|
|
|
</v-col>
|
|
|
|
</v-row>
|
|
|
|
<v-row justify="center" align="center">
|
|
|
|
<v-col cols="12">
|
2021-12-08 15:52:19 +08:00
|
|
|
<ProjectList :projects="projects" :list-options.sync="options" :display-author="true" />
|
2021-01-22 06:24:50 +08:00
|
|
|
</v-col>
|
|
|
|
</v-row>
|
|
|
|
</v-col>
|
|
|
|
|
2022-01-25 01:25:47 +08:00
|
|
|
<v-col cols="12" md="2" class="main-sidebar">
|
2021-12-04 01:28:35 +08:00
|
|
|
<br />
|
2021-02-13 19:34:36 +08:00
|
|
|
<HangarSponsor :sponsor="sponsor" />
|
2021-01-22 06:24:50 +08:00
|
|
|
|
2021-06-06 17:04:26 +08:00
|
|
|
<v-select />
|
2021-01-22 06:24:50 +08:00
|
|
|
|
2021-06-06 17:04:26 +08:00
|
|
|
<v-checkbox :label="$t('hangar.projectSearch.relevanceSort')" />
|
2021-01-22 06:24:50 +08:00
|
|
|
|
2021-12-04 01:28:35 +08:00
|
|
|
<v-list dense style="border-radius: 5px">
|
|
|
|
<v-row justify="center" align="center" style="position: relative; top: 5px; padding-bottom: 10px">
|
|
|
|
<v-subheader>Categories</v-subheader>
|
|
|
|
</v-row>
|
2021-06-07 13:25:56 +08:00
|
|
|
<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">
|
2021-01-22 06:24:50 +08:00
|
|
|
<v-list-item-icon>
|
|
|
|
<v-icon v-text="cat.icon" />
|
|
|
|
</v-list-item-icon>
|
|
|
|
<v-list-item-content>
|
2021-06-06 17:04:26 +08:00
|
|
|
<v-list-item-title v-text="$t(`project.category.${cat.apiName}`)" />
|
2021-01-23 09:44:41 +08:00
|
|
|
</v-list-item-content>
|
|
|
|
</v-list-item>
|
|
|
|
</v-list-item-group>
|
2021-01-22 06:24:50 +08:00
|
|
|
</v-list>
|
2021-12-04 01:28:35 +08:00
|
|
|
<br />
|
|
|
|
<v-list dense style="border-radius: 5px">
|
|
|
|
<v-row justify="center" align="center" style="position: relative; top: 5px; padding-bottom: 10px">
|
|
|
|
<v-subheader>Platforms</v-subheader>
|
|
|
|
</v-row>
|
2021-12-08 15:43:21 +08:00
|
|
|
<v-list-item-group v-model="filters.platforms" multiple>
|
|
|
|
<v-list-item v-for="platform in $store.getters.visiblePlatforms" :key="platform.enumName" :value="platform.name">
|
2021-01-22 06:24:50 +08:00
|
|
|
<v-list-item-icon>
|
2021-03-11 05:41:26 +08:00
|
|
|
<v-icon v-text="`$vuetify.icons.${platform.name.toLowerCase()}`" />
|
2021-01-22 06:24:50 +08:00
|
|
|
</v-list-item-icon>
|
|
|
|
<v-list-item-content>
|
2021-06-06 17:04:26 +08:00
|
|
|
<v-list-item-title v-text="platform.name" />
|
2021-03-11 05:41:26 +08:00
|
|
|
</v-list-item-content>
|
|
|
|
</v-list-item>
|
|
|
|
</v-list-item-group>
|
2021-01-22 06:24:50 +08:00
|
|
|
</v-list>
|
|
|
|
</v-col>
|
|
|
|
</v-row>
|
2021-12-04 01:28:35 +08:00
|
|
|
<br />
|
2021-01-22 06:24:50 +08:00
|
|
|
</div>
|
2021-01-21 11:36:18 +08:00
|
|
|
</template>
|
|
|
|
|
|
|
|
<script lang="ts">
|
2021-03-21 10:06:09 +08:00
|
|
|
import { Component } from 'nuxt-property-decorator';
|
2021-02-13 19:34:36 +08:00
|
|
|
import { PaginatedResult, Project, Sponsor } from 'hangar-api';
|
2021-03-17 01:25:32 +08:00
|
|
|
import { IPlatform } from 'hangar-internal';
|
2021-02-13 19:34:36 +08:00
|
|
|
import { Context } from '@nuxt/types';
|
2021-06-07 13:25:56 +08:00
|
|
|
import { DataOptions } from 'vuetify';
|
2021-03-20 15:24:17 +08:00
|
|
|
import { ProjectList } from '~/components/projects';
|
2021-02-04 17:34:24 +08:00
|
|
|
import HangarSponsor from '~/components/layouts/Sponsor.vue';
|
2021-03-11 05:41:26 +08:00
|
|
|
import { RootState } from '~/store';
|
2021-03-21 10:06:09 +08:00
|
|
|
import { HangarComponent } from '~/components/mixins';
|
2021-06-07 13:25:56 +08:00
|
|
|
import { Platform, ProjectCategory } from '~/types/enums';
|
2021-01-21 11:36:18 +08:00
|
|
|
|
2021-02-04 17:34:24 +08:00
|
|
|
@Component({
|
|
|
|
components: {
|
|
|
|
ProjectList,
|
|
|
|
HangarSponsor,
|
|
|
|
},
|
|
|
|
})
|
2021-03-21 10:06:09 +08:00
|
|
|
export default class Home extends HangarComponent {
|
2021-01-31 10:00:11 +08:00
|
|
|
// TODO implement filtering
|
2021-03-17 01:25:32 +08:00
|
|
|
projects!: PaginatedResult<Project>;
|
2021-02-13 19:34:36 +08:00
|
|
|
sponsor!: Sponsor;
|
2021-06-07 13:25:56 +08:00
|
|
|
filters = {
|
|
|
|
search: null as string | null,
|
|
|
|
platforms: [] as Platform[],
|
|
|
|
categories: [] as ProjectCategory[],
|
|
|
|
};
|
|
|
|
|
|
|
|
options = {
|
|
|
|
itemsPerPage: 25,
|
|
|
|
page: 1,
|
|
|
|
} as DataOptions;
|
2021-01-22 06:24:50 +08:00
|
|
|
|
2021-05-23 02:48:27 +08:00
|
|
|
head() {
|
2021-05-23 03:35:29 +08:00
|
|
|
const meta = this.$seo.head('Home', null, this.$route, null);
|
2021-08-05 01:22:13 +08:00
|
|
|
meta.script = meta.script ? meta.script : [];
|
2021-05-23 03:35:29 +08:00
|
|
|
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;
|
2021-03-11 05:41:26 +08:00
|
|
|
}
|
2021-01-21 11:36:18 +08:00
|
|
|
|
2021-06-07 13:25:56 +08:00
|
|
|
mounted() {
|
|
|
|
this.$watch('options', this.reloadProjectList, { deep: true });
|
|
|
|
this.$watch('filters', this.reloadProjectList, { deep: true });
|
|
|
|
}
|
|
|
|
|
2021-05-23 02:48:27 +08:00
|
|
|
get platforms(): IPlatform[] {
|
|
|
|
return Array.from((this.$store.state as RootState).platforms.values());
|
2021-01-22 04:32:08 +08:00
|
|
|
}
|
|
|
|
|
2021-06-07 13:25:56 +08:00
|
|
|
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,
|
2021-12-08 15:43:21 +08:00
|
|
|
platform: this.filters.platforms,
|
2021-06-07 13:25:56 +08:00
|
|
|
};
|
|
|
|
if (this.filters.search != null && this.filters.search.length > 0) {
|
|
|
|
requestOptions.q = this.filters.search;
|
|
|
|
}
|
|
|
|
return requestOptions;
|
|
|
|
}
|
|
|
|
|
2021-02-13 19:34:36 +08:00
|
|
|
async asyncData({ $api, $util }: Context) {
|
2021-03-17 01:25:32 +08:00
|
|
|
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] };
|
2021-01-21 11:36:18 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
</script>
|
2022-01-25 01:25:47 +08:00
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
@media (max-width: 746px) {
|
|
|
|
.main-plugin-content {
|
|
|
|
flex: 0 0 100%;
|
|
|
|
max-width: 100%;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@media (min-width: 747px) and (max-width: 840px) {
|
|
|
|
.main-sidebar {
|
|
|
|
max-width: 30%;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@media (min-width: 841px) and (max-width: 1370px) {
|
|
|
|
.main-sidebar {
|
|
|
|
max-width: 25%;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</style>
|