3c504e7048
* feat(frontend): ✨ add user registration page (WIP) * feat(backend): ✨ add user registration (WIP) * test(backend): ✅ add validator testing for registration schema * feat(backend): ✨ continued work on user sign-up * feat(backend): ✨ add signup flow and user/group settings * test(backend): ✅ user-creation tests and small refactor of existing tests * fix(backend): ✅ fix failing group tests * style: 🎨 fix lint issues Co-authored-by: hay-kot <hay-kot@pm.me>
269 lines
7.3 KiB
Vue
269 lines
7.3 KiB
Vue
<template>
|
|
<v-card :color="color" :dark="dark" flat :width="width" class="my-2">
|
|
<v-row>
|
|
<v-col v-for="(inputField, index) in items" :key="index" class="py-0" cols="12" sm="12">
|
|
<v-divider v-if="inputField.section" class="my-2" />
|
|
<v-card-title v-if="inputField.section" class="pl-0">
|
|
{{ inputField.section }}
|
|
</v-card-title>
|
|
<v-card-text v-if="inputField.sectionDetails" class="pl-0 mt-0 pt-0">
|
|
{{ inputField.sectionDetails }}
|
|
</v-card-text>
|
|
|
|
<!-- Check Box -->
|
|
<v-checkbox
|
|
v-if="inputField.type === fieldTypes.BOOLEAN"
|
|
v-model="value[inputField.varName]"
|
|
class="my-0 py-0"
|
|
:label="inputField.label"
|
|
:name="inputField.varName"
|
|
:hint="inputField.hint || ''"
|
|
@change="emitBlur"
|
|
/>
|
|
|
|
<!-- Text Field -->
|
|
<v-text-field
|
|
v-else-if="inputField.type === fieldTypes.TEXT"
|
|
v-model="value[inputField.varName]"
|
|
:readonly="inputField.fixed && updateMode"
|
|
filled
|
|
rounded
|
|
class="rounded-lg"
|
|
dense
|
|
:label="inputField.label"
|
|
:name="inputField.varName"
|
|
:hint="inputField.hint || ''"
|
|
:rules="[...rulesByKey(inputField.rules), ...defaultRules]"
|
|
lazy-validation
|
|
@blur="emitBlur"
|
|
/>
|
|
|
|
<!-- Text Area -->
|
|
<v-textarea
|
|
v-else-if="inputField.type === fieldTypes.TEXT_AREA"
|
|
v-model="value[inputField.varName]"
|
|
:readonly="inputField.fixed && updateMode"
|
|
filled
|
|
rounded
|
|
class="rounded-lg"
|
|
rows="3"
|
|
auto-grow
|
|
dense
|
|
:label="inputField.label"
|
|
:name="inputField.varName"
|
|
:hint="inputField.hint || ''"
|
|
:rules="[...rulesByKey(inputField.rules), ...defaultRules]"
|
|
lazy-validation
|
|
@blur="emitBlur"
|
|
/>
|
|
|
|
<!-- Option Select -->
|
|
<v-select
|
|
v-else-if="inputField.type === fieldTypes.SELECT"
|
|
v-model="value[inputField.varName]"
|
|
:readonly="inputField.fixed && updateMode"
|
|
filled
|
|
rounded
|
|
class="rounded-lg"
|
|
:prepend-icon="inputField.icons ? value[inputField.varName] : null"
|
|
:label="inputField.label"
|
|
:name="inputField.varName"
|
|
:items="inputField.options"
|
|
:return-object="false"
|
|
lazy-validation
|
|
@blur="emitBlur"
|
|
>
|
|
<template #item="{ item }">
|
|
<v-list-item-content>
|
|
<v-list-item-title>{{ item.text }}</v-list-item-title>
|
|
<v-list-item-subtitle>{{ item.description }}</v-list-item-subtitle>
|
|
</v-list-item-content>
|
|
</template>
|
|
</v-select>
|
|
|
|
<!-- Color Picker -->
|
|
<div v-else-if="inputField.type === fieldTypes.COLOR" class="d-flex" style="width: 100%">
|
|
<v-menu offset-y>
|
|
<template #activator="{ on }">
|
|
<v-btn class="my-2 ml-auto" style="min-width: 200px" :color="value[inputField.varName]" dark v-on="on">
|
|
{{ inputField.label }}
|
|
</v-btn>
|
|
</template>
|
|
<v-color-picker
|
|
v-model="value[inputField.varName]"
|
|
value="#7417BE"
|
|
hide-canvas
|
|
hide-inputs
|
|
show-swatches
|
|
class="mx-auto"
|
|
@input="emitBlur"
|
|
/>
|
|
</v-menu>
|
|
</div>
|
|
|
|
<div v-else-if="inputField.type === fieldTypes.OBJECT">
|
|
<base-auto-form
|
|
v-model="value[inputField.varName]"
|
|
:color="color"
|
|
:items="inputField.items"
|
|
@blur="emitBlur"
|
|
/>
|
|
</div>
|
|
|
|
<!-- List Type -->
|
|
<div v-else-if="inputField.type === fieldTypes.LIST">
|
|
<div v-for="(item, idx) in value[inputField.varName]" :key="idx">
|
|
<p>
|
|
{{ inputField.label }} {{ idx + 1 }}
|
|
<span>
|
|
<BaseButton class="ml-5" x-small delete @click="removeByIndex(value[inputField.varName], idx)" />
|
|
</span>
|
|
</p>
|
|
<v-divider class="mb-5 mx-2" />
|
|
<base-auto-form
|
|
v-model="value[inputField.varName][idx]"
|
|
:color="color"
|
|
:items="inputField.items"
|
|
@blur="emitBlur"
|
|
/>
|
|
</div>
|
|
<v-card-actions>
|
|
<v-spacer />
|
|
<BaseButton small @click="value[inputField.varName].push(getTemplate(inputField.items))"> New </BaseButton>
|
|
</v-card-actions>
|
|
</div>
|
|
</v-col>
|
|
</v-row>
|
|
</v-card>
|
|
</template>
|
|
|
|
<script>
|
|
import { ref } from "@nuxtjs/composition-api";
|
|
import { validators } from "@/composables/use-validators";
|
|
import { fieldTypes } from "@/composables/forms";
|
|
|
|
const BLUR_EVENT = "blur";
|
|
|
|
export default {
|
|
name: "BaseAutoForm",
|
|
props: {
|
|
value: {
|
|
default: null,
|
|
type: [Object, Array],
|
|
},
|
|
updateMode: {
|
|
default: false,
|
|
type: Boolean,
|
|
},
|
|
items: {
|
|
default: null,
|
|
type: Array,
|
|
},
|
|
width: {
|
|
type: [Number, String],
|
|
default: "max",
|
|
},
|
|
globalRules: {
|
|
default: null,
|
|
type: Array,
|
|
},
|
|
color: {
|
|
default: null,
|
|
type: String,
|
|
},
|
|
dark: {
|
|
default: false,
|
|
type: Boolean,
|
|
},
|
|
},
|
|
setup() {
|
|
const menu = ref({});
|
|
|
|
return {
|
|
menu,
|
|
fieldTypes,
|
|
validators,
|
|
};
|
|
},
|
|
computed: {
|
|
defaultRules() {
|
|
return this.rulesByKey(this.globalRules);
|
|
},
|
|
},
|
|
watch: {
|
|
items: {
|
|
immediate: true,
|
|
handler(val) {
|
|
// Initialize Value Object to Obtain all keys
|
|
if (!val) {
|
|
return;
|
|
}
|
|
for (let i = 0; i < val.length; i++) {
|
|
try {
|
|
if (this.value[val[i].varName]) {
|
|
continue;
|
|
}
|
|
} catch {}
|
|
|
|
if (val[i].type === "text" || val[i].type === "textarea") {
|
|
this.$set(this.value, val[i].varName, "");
|
|
} else if (val[i].type === "select") {
|
|
if (!val[i].options[0]) {
|
|
continue;
|
|
}
|
|
|
|
this.$set(this.value, val[i].varName, val[i].options[0].value);
|
|
} else if (val[i].type === "list") {
|
|
this.$set(this.value, val[i].varName, []);
|
|
} else if (val[i].type === "object") {
|
|
this.$set(this.value, val[i].varName, {});
|
|
} else if (val[i].type === "color") {
|
|
this.$set(this.value, val[i].varName, "");
|
|
this.$set(this.menu, val[i].varName, false);
|
|
}
|
|
}
|
|
},
|
|
},
|
|
},
|
|
methods: {
|
|
removeByIndex(list, index) {
|
|
// Removes the item at the index
|
|
list.splice(index, 1);
|
|
},
|
|
getTemplate(item) {
|
|
const obj = {};
|
|
|
|
item.forEach((field) => {
|
|
obj[field.varName] = "";
|
|
});
|
|
|
|
return obj;
|
|
},
|
|
rulesByKey(keys) {
|
|
const list = [];
|
|
|
|
if (keys === undefined) {
|
|
return list;
|
|
}
|
|
if (keys === null) {
|
|
return list;
|
|
}
|
|
if (keys === list) {
|
|
return list;
|
|
}
|
|
|
|
keys.forEach((key) => {
|
|
if (key in this.validators) {
|
|
list.push(this.validators[key]);
|
|
}
|
|
});
|
|
return list;
|
|
},
|
|
emitBlur() {
|
|
this.$emit(BLUR_EVENT, this.value);
|
|
},
|
|
},
|
|
};
|
|
</script>
|
|
|
|
<style lang="scss" scoped></style>
|