Files
cloudron-box/dashboard/src/components/DomainDialog.vue

150 lines
5.8 KiB
Vue

<script setup>
import { ref, useTemplateRef } from 'vue';
import { Dialog, TextInput, FormGroup, Checkbox } from 'pankow';
import DomainsModel from '../models/DomainsModel.js';
import DomainProviderForm from './DomainProviderForm.vue';
const emit = defineEmits([ 'success' ]);
const domainsModel = DomainsModel.create();
const dialog = useTemplateRef('dialog');
const busy = ref(false);
const errorMessage = ref('');
const editing = ref(false);
const domain = ref('');
const zoneName = ref('');
const provider = ref('');
const tlsProvider = ref('letsencrypt-prod-wildcard');
const showAdvanced = ref(false);
const customNameservers = ref(false);
const dnsConfig = ref(DomainsModel.createEmptyConfig());
const isFormValid = ref(false);
function checkValidity() {
isFormValid.value = form.value.checkValidity();
}
const form = useTemplateRef('form');
async function onSubmit() {
if (!form.value.reportValidity()) return;
busy.value = true;
errorMessage.value = '';
const config = dnsConfig.value;
config.customNameservers = customNameservers.value;
const tlsConfig = {
provider: tlsProvider.value,
wildcard: false
};
// UI uses -wildcard providers, API uses wildcard flag
if (tlsConfig.provider.indexOf('-wildcard') !== -1) {
tlsConfig.provider = tlsConfig.provider.replace('-wildcard', '');
tlsConfig.wildcard = true;
}
const func = editing.value ? domainsModel.update : domainsModel.add;
const [error] = await func(domain.value, zoneName.value, provider.value, config, null, tlsConfig);
if (error) {
errorMessage.value = error.body ? error.body.message : 'Internal error';
busy.value = false;
return console.error(error);
}
emit('success');
dialog.value.close();
}
defineExpose({
open(d) {
d = d || { config: {}, tlsConfig: {}};
provider.value = d.provider || '';
dnsConfig.value = d.config;
tlsProvider.value = d.tlsConfig.provider || 'letsencrypt-prod-wildcard';
busy.value = false;
showAdvanced.value = false;
errorMessage.value = '';
editing.value = !!d.domain;
domain.value = d.domain || '';
zoneName.value = d.zoneName || '';
customNameservers.value = d.config.customNameservers;
dialog.value.open();
// ensure we trigger this once
setTimeout(checkValidity, 100);
}
});
</script>
<template>
<Dialog ref="dialog"
:title="editing ? $t('domains.domainDialog.editTitle', { domain: domain }) : $t('domains.domainDialog.addTitle')"
:modal="busy"
:confirm-busy="busy"
:confirm-active="!busy && isFormValid"
:confirm-label="$t('main.dialog.save')"
:reject-label="busy ? null : $t('main.dialog.cancel')"
reject-style="secondary"
@confirm="onSubmit()"
>
<p class="text-danger" v-show="errorMessage">{{ errorMessage }}</p>
<form ref="form" @submit.prevent="onSubmit()" autocomplete="off" @input="checkValidity()">
<fieldset :disabled="busy">
<input style="display: none;" type="submit" :disabled="busy"/>
<div class="error-label" v-show="error">{{ error }}</div>
<FormGroup>
<label for="domainInput">{{ $t('domains.domainDialog.domain') }}</label>
<TextInput id="domainInput" v-model="domain" placeholder="example.com" :readonly="editing ? true : undefined" required />
</FormGroup>
<DomainProviderForm v-model:provider="provider" v-model:dns-config="dnsConfig" v-model:tls-provider="tlsProvider" :domain="domain" :show-advanced="showAdvanced" />
<p style="margin-top: 15px" v-show="!showAdvanced" @click="showAdvanced = true" class="actionable">{{ $t('domains.domainDialog.advancedAction') }}</p>
<div v-show="showAdvanced">
<FormGroup>
<label for="zoneNameInput">{{ $t('domains.domainDialog.zoneName') }} <sup><a href="https://docs.cloudron.io/domains/#zone-name" class="help" target="_blank"><i class="fa fa-question-circle"></i></a></sup></label>
<TextInput id="zoneNameInput" v-model="zoneName" />
</FormGroup>
<Checkbox v-model="customNameservers" :label="$t('domains.domainDialog.customNameservers')" />
<!-- custom certificate -->
<div v-if="tlsProvider === 'fallback'">
<label >{{ $t('domains.domainDialog.fallbackCertCustomCert') }}</label>
<p v-html="$t('domains.domainDialog.fallbackCertCustomCertInfo', { customCertLink: 'https://docs.cloudron.io/certificates/#custom-certificates' })"></p>
</div>
<FormGroup v-if="tlsProvider === 'fallback'">
<div class="input-group">
<input type="file" id="fallbackCertFileInput" style="display:none"/>
<input type="text" class="form-control" :placeholder="$t('domains.domainDialog.fallbackCertCertificatePlaceholder')" ng-model="domainConfigure.fallbackCert.certificateFileName" name="cert" onclick="getElementById('fallbackCertFileInput').click();" style="cursor: pointer;" ng-disabled="domainConfigure.busy">
<span class="input-group-addon"><i class="fa fa-upload" onclick="getElementById('fallbackCertFileInput').click();"></i></span>
</div>
</FormGroup>
<FormGroup v-if="tlsProvider === 'fallback'">
<div class="input-group">
<input type="file" id="fallbackKeyFileInput" style="display:none"/>
<input type="text" class="form-control" :placeholder="$t('domains.domainDialog.fallbackCertKeyPlaceholder')" ng-model="domainConfigure.fallbackCert.keyFileName" id="fallbackKeyInput" name="key" onclick="getElementById('fallbackKeyFileInput').click();" style="cursor: pointer;" ng-disabled="domainConfigure.busy">
<span class="input-group-addon"><i class="fa fa-upload" onclick="getElementById('fallbackKeyFileInput').click();"></i></span>
</div>
</FormGroup>
</div>
</fieldset>
</form>
</Dialog>
</template>