mirror of
https://github.com/misskey-dev/misskey.git
synced 2026-03-21 03:30:42 +00:00
127 lines
5.4 KiB
Vue
127 lines
5.4 KiB
Vue
<!--
|
|
SPDX-FileCopyrightText: syuilo and misskey-project
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
-->
|
|
|
|
<template>
|
|
<div v-if="Object.values(form).filter(item => typeof item.hidden !== 'boolean' || item.hidden === true).length > 0" class="_gaps_m">
|
|
<template v-for="v, k in form">
|
|
<template v-if="typeof v.hidden == 'function' ? v.hidden(values) : v.hidden"></template>
|
|
<MkInput v-else-if="v.type === 'number'" v-model="values[k]" type="number" :step="v.step || 1" :manualSave="v.manualSave" @savingStateChange="(changed, invalid) => onSavingStateChange(k, changed, invalid)">
|
|
<template #label><span v-text="v.label || k"></span><span v-if="v.required === false"> ({{ i18n.ts.optional }})</span></template>
|
|
<template v-if="v.description" #caption>{{ v.description }}</template>
|
|
</MkInput>
|
|
<MkInput v-else-if="v.type === 'string' && !v.multiline" v-model="values[k]" type="text" :mfmAutocomplete="v.treatAsMfm" :manualSave="v.manualSave" @savingStateChange="(changed, invalid) => onSavingStateChange(k, changed, invalid)">
|
|
<template #label><span v-text="v.label || k"></span><span v-if="v.required === false"> ({{ i18n.ts.optional }})</span></template>
|
|
<template v-if="v.description" #caption>{{ v.description }}</template>
|
|
</MkInput>
|
|
<MkTextarea v-else-if="v.type === 'string' && v.multiline" v-model="values[k]" :mfmAutocomplete="v.treatAsMfm" :mfmPreview="v.treatAsMfm" :manualSave="v.manualSave" @savingStateChange="(changed, invalid) => onSavingStateChange(k, changed, invalid)">
|
|
<template #label><span v-text="v.label || k"></span><span v-if="v.required === false"> ({{ i18n.ts.optional }})</span></template>
|
|
<template v-if="v.description" #caption>{{ v.description }}</template>
|
|
</MkTextarea>
|
|
<MkSwitch v-else-if="v.type === 'boolean'" v-model="values[k]">
|
|
<span v-text="v.label || k"></span>
|
|
<template v-if="v.description" #caption>{{ v.description }}</template>
|
|
</MkSwitch>
|
|
<MkSelect v-else-if="v.type === 'enum'" v-model="values[k]" :items="getMkSelectDef(v)">
|
|
<template #label><span v-text="v.label || k"></span><span v-if="v.required === false"> ({{ i18n.ts.optional }})</span></template>
|
|
</MkSelect>
|
|
<MkRadios v-else-if="v.type === 'radio'" v-model="values[k]" :options="getRadioOptionsDef(v)">
|
|
<template #label><span v-text="v.label || k"></span><span v-if="v.required === false"> ({{ i18n.ts.optional }})</span></template>
|
|
</MkRadios>
|
|
<MkRange v-else-if="v.type === 'range'" v-model="values[k]" :min="v.min" :max="v.max" :step="v.step" :textConverter="v.textConverter">
|
|
<template #label><span v-text="v.label || k"></span><span v-if="v.required === false"> ({{ i18n.ts.optional }})</span></template>
|
|
<template v-if="v.description" #caption>{{ v.description }}</template>
|
|
</MkRange>
|
|
<MkButton v-else-if="v.type === 'button'" @click="v.action($event, values)">
|
|
<span v-text="v.content || k"></span>
|
|
</MkButton>
|
|
<XFile
|
|
v-else-if="v.type === 'drive-file'"
|
|
:fileId="v.defaultFileId"
|
|
:validate="async f => !v.validate || await v.validate(f)"
|
|
@update="f => values[k] = f"
|
|
/>
|
|
</template>
|
|
</div>
|
|
<MkResult v-else type="empty" :text="i18n.ts.nothingToConfigure"/>
|
|
</template>
|
|
|
|
<script lang="ts" setup>
|
|
import { computed, ref, watch } from 'vue';
|
|
import XFile from '@/components/MkForm.file.vue';
|
|
import MkInput from '@/components/MkInput.vue';
|
|
import MkTextarea from '@/components/MkTextarea.vue';
|
|
import MkSwitch from '@/components/MkSwitch.vue';
|
|
import MkSelect from '@/components/MkSelect.vue';
|
|
import MkRange from '@/components/MkRange.vue';
|
|
import MkButton from '@/components/MkButton.vue';
|
|
import MkRadios from '@/components/MkRadios.vue';
|
|
import { i18n } from '@/i18n.js';
|
|
import type { MkSelectItem } from '@/components/MkSelect.vue';
|
|
import type { MkRadiosOption } from '@/components/MkRadios.vue';
|
|
import type { Form, EnumFormItem, RadioFormItem } from '@/utility/form.js';
|
|
import { items } from 'happy-dom/lib/PropertySymbol';
|
|
|
|
const props = defineProps<{
|
|
form: Form;
|
|
}>();
|
|
|
|
const emit = defineEmits<{
|
|
(ev: 'canSaveStateChange', canSave: boolean): void;
|
|
}>();
|
|
|
|
// TODO: ジェネリックにしたい
|
|
const values = defineModel<Record<string, any>>({ required: true });
|
|
|
|
// 保存可能状態の管理
|
|
const inputSavingStates = ref<Record<string, { changed: boolean; invalid: boolean }>>({});
|
|
|
|
function onSavingStateChange(key: string, changed: boolean, invalid: boolean) {
|
|
inputSavingStates.value[key] = { changed, invalid };
|
|
}
|
|
|
|
const canSave = computed(() => {
|
|
for (const key in inputSavingStates.value) {
|
|
const state = inputSavingStates.value[key];
|
|
if (
|
|
('manualSave' in props.form[key] && props.form[key].manualSave && state.changed) ||
|
|
state.invalid
|
|
) {
|
|
return false;
|
|
}
|
|
if ('required' in props.form[key] && props.form[key].required) {
|
|
const val = values.value[key];
|
|
if (val === null || val === undefined || val === '') {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
return true;
|
|
});
|
|
|
|
watch(canSave, (newCanSave) => {
|
|
emit('canSaveStateChange', newCanSave);
|
|
}, { immediate: true });
|
|
|
|
function getMkSelectDef(def: EnumFormItem): MkSelectItem[] {
|
|
return def.enum.map((v) => {
|
|
if (typeof v === 'string') {
|
|
return { value: v, label: v };
|
|
} else {
|
|
return { value: v.value, label: v.label };
|
|
}
|
|
});
|
|
}
|
|
|
|
function getRadioOptionsDef(def: RadioFormItem): MkRadiosOption[] {
|
|
return def.options.map<MkRadiosOption>((v) => {
|
|
if (typeof v === 'string') {
|
|
return { value: v, label: v };
|
|
} else {
|
|
return { value: v.value, label: v.label };
|
|
}
|
|
});
|
|
}
|
|
</script>
|