funquail/front/src/components/admin/SettingsGroup.vue
2022-09-06 09:26:36 +00:00

251 lines
6.8 KiB
Vue

<template>
<form
:id="group.id"
class="ui form component-settings-group"
@submit.prevent="save"
>
<div class="ui divider" />
<h3 class="ui header">
{{ group.label }}
</h3>
<div
v-if="errors.length > 0"
role="alert"
class="ui negative message"
>
<h4 class="header">
<translate translate-context="Content/*/Error message.Title">
Error while saving settings
</translate>
</h4>
<ul class="list">
<li
v-for="(error, key) in errors"
:key="key"
>
{{ error }}
</li>
</ul>
</div>
<div
v-if="result"
class="ui positive message"
>
<translate translate-context="Content/Settings/Paragraph">
Settings updated successfully.
</translate>
</div>
<p v-if="group.help">
{{ group.help }}
</p>
<div
v-for="(setting, key) in settings"
:key="key"
class="ui field"
>
<template v-if="setting.field.widget.class !== 'CheckboxInput'">
<label :for="setting.identifier">{{ setting.verbose_name }}</label>
<p v-if="setting.help_text">
{{ setting.help_text }}
</p>
</template>
<content-form
v-if="setting.fieldType === 'markdown'"
v-model="values[setting.identifier]"
v-bind="setting.fieldParams"
/>
<signup-form-builder
v-else-if="setting.fieldType === 'formBuilder'"
v-model="values[setting.identifier]"
:signup-approval-enabled="values.moderation__signup_approval_enabled"
/>
<input
v-else-if="setting.field.widget.class === 'PasswordInput'"
:id="setting.identifier"
v-model="values[setting.identifier]"
:name="setting.identifier"
type="password"
class="ui input"
>
<input
v-else-if="setting.field.widget.class === 'TextInput'"
:id="setting.identifier"
v-model="values[setting.identifier]"
:name="setting.identifier"
type="text"
class="ui input"
>
<input
v-else-if="setting.field.class === 'IntegerField'"
:id="setting.identifier"
v-model.number="values[setting.identifier]"
:name="setting.identifier"
type="number"
class="ui input"
>
<textarea
v-else-if="setting.field.widget.class === 'Textarea'"
:id="setting.identifier"
v-model="values[setting.identifier]"
:name="setting.identifier"
type="text"
class="ui input"
/>
<div
v-else-if="setting.field.widget.class === 'CheckboxInput'"
class="ui toggle checkbox"
>
<input
:id="setting.identifier"
v-model="values[setting.identifier]"
:name="setting.identifier"
type="checkbox"
>
<label :for="setting.identifier">{{ setting.verbose_name }}</label>
<p v-if="setting.help_text">
{{ setting.help_text }}
</p>
</div>
<select
v-else-if="setting.field.class === 'MultipleChoiceField'"
:id="setting.identifier"
v-model="values[setting.identifier]"
multiple
class="ui search selection dropdown"
>
<option
v-for="(v, index) in setting.additional_data.choices"
:key="index"
:value="v[0]"
>
{{ v[1] }}
</option>
</select>
<div v-else-if="setting.field.widget.class === 'ImageWidget'">
<input
:id="setting.identifier"
:ref="setting.identifier"
type="file"
>
<div v-if="values[setting.identifier]">
<div class="ui hidden divider" />
<h3 class="ui header">
<translate translate-context="Content/Settings/Title/Noun">
Current image
</translate>
</h3>
<img
v-if="values[setting.identifier]"
class="ui image"
alt=""
:src="$store.getters['instance/absoluteUrl'](values[setting.identifier])"
>
</div>
</div>
</div>
<button
type="submit"
:class="['ui', {'loading': isLoading}, 'right', 'floated', 'success', 'button']"
>
<translate translate-context="Content/*/Button.Label/Verb">
Save
</translate>
</button>
</form>
</template>
<script>
import axios from 'axios'
import { cloneDeep } from 'lodash-es'
import SignupFormBuilder from '~/components/admin/SignupFormBuilder.vue'
import useFormData from '~/composables/useFormData'
export default {
components: {
SignupFormBuilder
},
props: {
group: { type: Object, required: true },
settingsData: { type: Array, required: true }
},
data () {
return {
values: {},
result: null,
errors: [],
isLoading: false
}
},
computed: {
settings () {
const byIdentifier = {}
this.settingsData.forEach(e => {
byIdentifier[e.identifier] = e
})
return this.group.settings.map(e => {
return { ...byIdentifier[e.name], fieldType: e.fieldType, fieldParams: e.fieldParams || {} }
})
},
fileSettings () {
return this.settings.filter((s) => {
return s.field.widget.class === 'ImageWidget'
})
}
},
created () {
const self = this
this.settings.forEach(e => {
self.values[e.identifier] = e.value
})
},
methods: {
save () {
const self = this
this.isLoading = true
self.errors = []
self.result = null
let postData = self.values
let contentType = 'application/json'
const fileSettingsIDs = this.fileSettings.map((s) => { return s.identifier })
if (fileSettingsIDs.length > 0) {
const data = this.settings.reduce((data, setting) => {
if (fileSettingsIDs.includes(setting.identifier)) {
const [input] = this.$refs[setting.identifier]
const { files } = input
// TODO (wvffle): Move to the top of setup
const logger = useLogger()
logger.debug('ref', input, files)
if (files?.length > 0) {
data[s.identifier] = files[0]
}
} else {
postData.append(s.identifier, this.values[s.identifier])
}
return data
}, {})
contentType = 'multipart/form-data'
postData = useFormData(data)
}
axios.post('instance/admin/settings/bulk/', postData, {
headers: {
'Content-Type': contentType
}
}).then((response) => {
self.result = true
response.data.forEach((s) => {
self.values[s.identifier] = s.value
})
self.isLoading = false
self.$store.dispatch('instance/fetchSettings')
}, error => {
self.isLoading = false
self.errors = error.backendErrors
})
}
}
}
</script>