Hangar/frontend/pages/new.vue

366 lines
16 KiB
Vue
Raw Normal View History

2021-01-22 01:19:00 +08:00
<template>
2021-02-05 14:51:51 +08:00
<v-stepper v-model="step">
<v-stepper-header>
<v-stepper-step step="1" :complete="step > 1">{{ $t('project.new.step1.title') }}</v-stepper-step>
<v-divider />
2021-02-06 03:50:18 +08:00
<v-stepper-step step="2" :complete="step > 2 && forms.step2" :rules="[() => noBasicSettingsError]"
2021-02-05 14:51:51 +08:00
>{{ $t('project.new.step2.title') }}<small v-show="!noBasicSettingsError">Missing Information</small></v-stepper-step
>
<v-divider />
<v-stepper-step step="3" :complete="step > 3"
>{{ $t('project.new.step3.title') }}<small>{{ $t('project.new.step3.optional') }}</small></v-stepper-step
>
<v-divider />
<v-stepper-step step="4" :complete="step > 4"
>{{ $t('project.new.step4.title') }}<small>{{ $t('project.new.step4.optional') }}</small></v-stepper-step
>
<v-divider />
2021-02-06 03:50:18 +08:00
<v-stepper-step step="5" :complete="!projectLoading">{{ $t('project.new.step5.title') }}</v-stepper-step>
2021-02-05 14:51:51 +08:00
</v-stepper-header>
<v-stepper-items>
<StepperStepContent :step="1" @back="$router.push('/')" @continue="step = 2">
<v-card class="mb-12 pa-1" min-height="200px">
<v-card-title v-if="$vuetify.breakpoint.smAndDown">
{{ $t('project.new.step1.title') }}
</v-card-title>
2021-02-11 19:48:52 +08:00
<!-- eslint-disable-next-line vue/no-v-html -->
2021-02-05 14:51:51 +08:00
<v-card-text v-html="$t('project.new.step1.text')"></v-card-text>
</v-card>
</StepperStepContent>
<StepperStepContent :step="2" :allow-continue="noBasicSettingsError" @back="step = 1" @continue="step = 3">
<v-card max-width="800" class="mx-auto">
<v-card-title v-if="$vuetify.breakpoint.smAndDown">
{{ $t('project.new.step2.title') }}
</v-card-title>
2021-02-06 03:50:18 +08:00
<v-form v-model="forms.step2">
2021-02-05 14:51:51 +08:00
<v-container>
<v-row justify="space-around">
<v-col cols="12" md="6">
<v-select
v-model="form.ownerId"
:items="projectOwners"
dense
filled
item-text="name"
item-value="userId"
:label="$t('project.new.step2.userselect')"
2021-02-12 13:44:49 +08:00
:rules="[$util.$vc.require('Project owner')]"
2021-02-05 14:51:51 +08:00
:append-icon="createAsIcon"
/>
</v-col>
<v-col cols="12" md="6">
2021-02-11 19:48:52 +08:00
<!-- todo custom rule to check if a name exist already -->
2021-02-05 14:51:51 +08:00
<v-text-field
v-model.trim="form.name"
autofocus
dense
filled
2021-02-12 13:44:49 +08:00
:error-messages="nameErrors"
2021-02-05 14:51:51 +08:00
:label="$t('project.new.step2.projectname')"
:rules="[$util.$vc.require('Name')]"
append-icon="mdi-form-textbox"
/>
</v-col>
<v-col cols="12" md="8">
<v-text-field
v-model.trim="form.description"
dense
filled
clearable
:label="$t('project.new.step2.projectsummary')"
:rules="[$util.$vc.require('Description')]"
append-icon="mdi-card-text"
/>
</v-col>
<v-col cols="12" md="4">
<v-select
v-model="form.category"
:append-icon="categoryIcon"
:items="$store.getters.visibleCategories"
dense
filled
:label="$t('project.new.step2.projectcategory')"
item-text="title"
item-value="apiName"
:rules="[$util.$vc.require('Category')]"
/>
</v-col>
</v-row>
</v-container>
</v-form>
</v-card>
</StepperStepContent>
<StepperStepContent :step="3" @back="step = 2" @continue="step = 4">
<v-card max-width="800" class="mx-auto">
<v-card-title v-if="$vuetify.breakpoint.smAndDown">
{{ $t('project.new.step3.title') }}
</v-card-title>
2021-02-06 03:50:18 +08:00
<v-container>
<div class="text-h6 pt-1">
<v-icon color="info" large style="transform: rotate(-45deg)" class="mb-1">mdi-link</v-icon>
{{ $t('project.new.step3.links') }}
</div>
<v-divider class="mb-2" />
<v-row>
<v-col cols="12">
<v-text-field
v-model.trim="form.links.homepage"
dense
hide-details
filled
:label="$t('project.new.step3.homepage')"
append-icon="mdi-home-search"
/>
</v-col>
<v-col cols="12">
<v-text-field
v-model.trim="form.links.issues"
dense
hide-details
filled
:label="$t('project.new.step3.issues')"
append-icon="mdi-bug"
/>
</v-col>
<v-col cols="12">
<v-text-field
v-model.trim="form.links.source"
dense
hide-details
filled
:label="$t('project.new.step3.source')"
append-icon="mdi-source-branch"
/>
</v-col>
<v-col cols="12">
<v-text-field
v-model.trim="form.links.support"
dense
hide-details
filled
:label="$t('project.new.step3.support')"
append-icon="mdi-face-agent"
/>
</v-col>
</v-row>
<div class="text-h6 pt-5">
<v-icon color="info" large class="mb-1">mdi-license</v-icon>
{{ $t('project.new.step3.licence') }}
</div>
<v-divider class="mb-2" />
<v-row>
<v-col cols="12" :md="isCustomLicense ? 4 : 6">
<v-select
v-model="form.license.type"
dense
hide-details
filled
clearable
:items="licences"
:label="$t('project.new.step3.type')"
/>
</v-col>
<v-col v-if="isCustomLicense" cols="12" md="8">
<v-text-field v-model.trim="form.license.customName" dense hide-details filled :label="$t('project.new.step3.customName')" />
</v-col>
<v-col cols="12" :md="isCustomLicense ? 12 : 6">
<v-text-field v-model.trim="form.license.url" dense hide-details filled :label="$t('project.new.step3.url')" />
</v-col>
</v-row>
<div class="text-h6 pt-5">
<v-icon color="info" large class="mb-1">mdi-cloud-search</v-icon>
{{ $t('project.new.step3.seo') }}
</div>
<v-divider class="mb-2" />
<v-row>
<v-col cols="12">
<v-combobox
v-model="form.keywords"
small-chips
deletable-chips
multiple
dense
hide-details
filled
:delimiters="[' ', ',', '.']"
:label="$t('project.new.step3.keywords')"
append-icon="mdi-file-word-box"
/>
</v-col>
</v-row>
</v-container>
2021-02-05 14:51:51 +08:00
</v-card>
</StepperStepContent>
2021-02-06 03:50:18 +08:00
<StepperStepContent
:step="4"
@back="step = 3"
@continue="
step = 5;
createProject();
"
>
2021-02-05 14:51:51 +08:00
<v-tabs v-model="spigotConvertTab" fixed-tabs>
<v-tab v-text="$t('project.new.step4.convert')"></v-tab>
<v-tab v-text="$t('project.new.step4.preview')"></v-tab>
<v-tab v-text="$t('project.new.step4.tutorial')"></v-tab>
</v-tabs>
<v-tabs-items v-model="spigotConvertTab">
<!-- todo spigot bbcode converter thingy -->
<v-tab-item>1 </v-tab-item>
<v-tab-item>2 </v-tab-item>
<v-tab-item>3 </v-tab-item>
</v-tabs-items>
</StepperStepContent>
<StepperStepContent :step="5" hide-buttons>
<v-card>
<v-card-text class="text-center">
2021-02-06 03:50:18 +08:00
<v-progress-circular v-if="projectLoading" indeterminate color="red" size="50"></v-progress-circular>
<div v-if="!projectError" class="text-h5 mt-2">{{ $t('project.new.step5.text') }}</div>
<template v-else>
<div class="text-h5 mt-2">{{ $t('project.new.error') }}</div>
2021-02-06 06:54:07 +08:00
<v-btn
@click="
step = 1;
projectLoading = true;
projectError = false;
"
>Retry</v-btn
>
2021-02-06 03:50:18 +08:00
</template>
2021-02-05 14:51:51 +08:00
</v-card-text>
</v-card>
</StepperStepContent>
</v-stepper-items>
2021-01-31 01:27:36 +08:00
</v-stepper>
2021-01-22 01:19:00 +08:00
</template>
<script lang="ts">
2021-02-12 13:44:49 +08:00
import { Component, Vue, Watch } from 'nuxt-property-decorator';
2021-02-05 14:51:51 +08:00
import { Context } from '@nuxt/types';
2021-02-07 13:44:53 +08:00
import { ProjectOwner } from 'hangar-internal';
2021-02-12 13:44:49 +08:00
import { AxiosError } from 'axios';
2021-02-05 14:51:51 +08:00
import StepperStepContent from '~/components/steppers/StepperStepContent.vue';
import { RootState } from '~/store';
import { ProjectCategory } from '~/types/enums';
interface NewProjectForm {
ownerId: ProjectOwner['userId'];
name: string;
description: string;
category: ProjectCategory;
pageContent: string | null;
links: {
homepage: string | null;
issues: string | null;
source: string | null;
support: string | null;
};
license: {
type: string | null;
url: string | null;
customName: string | null;
};
keywords: string[];
}
2021-01-31 10:00:11 +08:00
2021-01-31 01:27:36 +08:00
@Component({
2021-02-05 14:51:51 +08:00
components: {
StepperStepContent,
},
head: {
title: 'New Project',
},
2021-01-31 01:27:36 +08:00
})
export default class NewPage extends Vue {
step = 1;
spigotConvertTab = 0;
2021-02-06 03:50:18 +08:00
projectLoading = true;
projectError = false;
projectOwners!: ProjectOwner[];
error = null as string | null;
2021-02-05 14:51:51 +08:00
form = ({
category: ProjectCategory.ADMIN_TOOLS,
links: {},
license: {},
keywords: [],
} as unknown) as NewProjectForm;
2021-02-12 13:44:49 +08:00
nameErrors: string[] = [];
2021-02-05 14:51:51 +08:00
forms = {
2021-02-06 03:50:18 +08:00
step2: false,
2021-02-05 14:51:51 +08:00
};
get categoryIcon() {
return (this.$store.state as RootState).projectCategories.get(this.form.category)?.icon;
}
2021-01-31 01:27:36 +08:00
2021-02-05 14:51:51 +08:00
get createAsIcon() {
return this.projectOwners.find((po) => po.userId === this.form.ownerId)?.isOrganization ? 'mdi-account-multiple' : 'mdi-account';
2021-01-31 01:27:36 +08:00
}
2021-02-05 14:51:51 +08:00
get isCustomLicense() {
return this.form.license.type === '(custom)';
2021-01-31 01:27:36 +08:00
}
2021-01-22 01:19:00 +08:00
2021-02-05 14:51:51 +08:00
get noBasicSettingsError() {
2021-02-06 03:50:18 +08:00
return this.step !== 2 || this.forms.step2;
2021-02-05 14:51:51 +08:00
}
// TODO do we want to get those from the server? Jake: I think so, it'd be nice to admins to be able to configure default licenses, but not needed for MVP
2021-01-31 01:27:36 +08:00
get licences() {
2021-02-05 14:51:51 +08:00
return ['MIT', 'Apache 2.0', 'GPL', 'LGPL', '(custom)'];
2021-01-31 01:27:36 +08:00
}
2021-01-31 02:50:12 +08:00
2021-02-05 14:51:51 +08:00
async asyncData({ $api }: Context) {
return {
projectOwners: await $api.requestInternal<ProjectOwner[]>('projects/possibleOwners'),
};
}
created() {
this.form.ownerId = this.projectOwners.find((po) => !po.isOrganization)?.userId!;
}
2021-01-22 01:19:00 +08:00
2021-02-06 03:50:18 +08:00
createProject() {
console.log(this.form);
this.$api
2021-02-06 15:45:13 +08:00
.requestInternal<string>('projects/create', true, 'post', this.form)
.then((url) => {
this.$router.push(url);
2021-02-06 03:50:18 +08:00
})
.catch((err) => {
this.projectError = true;
2021-02-10 12:40:34 +08:00
this.$util.handleRequestError(err, 'Unable to create project');
2021-02-06 03:50:18 +08:00
})
.finally(() => {
this.projectLoading = false;
});
}
2021-02-12 13:44:49 +08:00
// This is very useful. Prob should have a generalization of this that works elsewhere. I didn't make it a rule because it relies on other input (the ownerId)
@Watch('form.name')
onProjectNameChange(val: string) {
if (!val) {
this.nameErrors = [];
return;
}
this.$api
.requestInternal('projects/validateName', false, 'get', {
userId: this.form.ownerId,
value: val,
})
.then(() => {
this.nameErrors = [];
})
.catch((err: AxiosError) => {
this.nameErrors = [];
if (!err.response?.data.isHangarApiException) {
return this.$util.handleRequestError(err);
}
this.nameErrors.push(err.response.data.message);
});
}
2021-01-31 01:27:36 +08:00
}
2021-02-05 14:51:51 +08:00
</script>