Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Port mail admin settings to vue #35460

Closed
wants to merge 1 commit into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
45 changes: 19 additions & 26 deletions apps/settings/lib/Settings/Admin/Mail.php
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
<?php
declare(strict_types=1);
/**
* @copyright Copyright (c) 2016 Arthur Schiwon <[email protected]>
*
Expand All @@ -8,6 +9,7 @@
* @author Julius Härtl <[email protected]>
* @author Lukas Reschke <[email protected]>
* @author Roeland Jago Douma <[email protected]>
* @author Carl Schwan <[email protected]>
*
* @license GNU AGPL version 3 or any later version
*
Expand All @@ -28,30 +30,26 @@
namespace OCA\Settings\Settings\Admin;

use OCP\AppFramework\Http\TemplateResponse;
use OCP\AppFramework\Services\IInitialState;
use OCP\IConfig;
use OCP\IL10N;
use OCP\Settings\IDelegatedSettings;
use OCP\IURLGenerator;

class Mail implements IDelegatedSettings {
/** @var IConfig */
private $config;
private IConfig $config;
private IL10N $l;
private IInitialState $initialState;
private IURLGenerator $urlGenerator;

/** @var IL10N $l */
private $l;

/**
* @param IConfig $config
* @param IL10N $l
*/
public function __construct(IConfig $config, IL10N $l) {
public function __construct(IConfig $config, IL10N $l, IInitialState $initialState, IURLGenerator $urlGenerator) {
$this->config = $config;
$this->l = $l;
$this->initialState = $initialState;
$this->urlGenerator = $urlGenerator;
}

/**
* @return TemplateResponse
*/
public function getForm() {
public function getForm(): TemplateResponse {
$parameters = [
// Mail
'sendmail_is_available' => (bool) \OC_Helper::findBinaryPath('sendmail'),
Expand All @@ -76,24 +74,19 @@ public function getForm() {
$parameters['mail_smtpmode'] = 'smtp';
}

foreach ($parameters as $key => $parameter) {
$this->initialState->provideInitialState($key, $parameter);
}
$this->initialState->provideInitialState('emailAdminDocUrl', 'https://jroiere.com'); //$this->urlGenerator->linkToDocs('admin-email'));

return new TemplateResponse('settings', 'settings/admin/additional-mail', $parameters, '');
}

/**
* @return string the section ID, e.g. 'sharing'
*/
public function getSection() {
public function getSection(): string {
return 'server';
}

/**
* @return int whether the form should be rather on the top or bottom of
* the admin section. The forms are arranged in ascending order of the
* priority values. It is required to return a value between 0 and 100.
*
* E.g.: 70
*/
public function getPriority() {
public function getPriority(): int {
return 10;
}

Expand Down
263 changes: 263 additions & 0 deletions apps/settings/src/components/MailDeliverySettings.vue
Original file line number Diff line number Diff line change
@@ -0,0 +1,263 @@
<!--
- @copyright 2022 Carl Schwan <[email protected]>
-
- @author Carl Schwan <[email protected]>
-
- @license GNU AGPL version 3 or any later version
-
- This program is free software: you can redistribute it and/or modify
- it under the terms of the GNU Affero General Public License as
- published by the Free Software Foundation, either version 3 of the
- License, or (at your option) any later version.
-
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU Affero General Public License for more details.
-
- You should have received a copy of the GNU Affero General Public License
- along with this program. If not, see <http://www.gnu.org/licenses/>.
-
-->

<template>
<NcSettingsSection :title="t('settings', 'Email server')"
:description="t('settings', 'It is important to set up this server to be able to send emails, like for password reset and notifications.')"
:doc-url="adminDocUrl"
class="spacing">

<div>
<div class="form-control">
<label for="mail_smtpmode">{{ t('settings', 'Send mode') }}</label>
<select name="mail_smtpmode" id="mail_smtpmode">
<option v-for="smtpMode in smtpModes"
:key="smtpMode[0]"
:value="smtpMode[0]"
:selected="smtpMode[0] === mailSmtpMode">
{{ smtpMode[1] }}
</option>
</select>
</div>

<div v-if="mailSmtpMode === 'smtp'" class="form-control">
<label id="mail_smtpsecure_label" for="mail_smtpsecure">
{{ t('settings', 'Encryption') }}
</label>
<select name="mail_smtpsecure" id="mail_smtpsecure">
<option v-for="(secure, name) in smtpSecure"
:key="secure"
:value="secure"
:selected="secure === mailSmtpSecure">
{{ smtpSecure[name] }}
</option>
</select>
</div>

<div v-if="mailSmtpMode === 'sendmail'" class="form-control">
<label for="mail_sendmailmode">
{{ t('settings', 'Sendmail mode') }}
</label>
<select name="mail_sendmailmode" id="mail_sendmailmode">
<option v-for="(sendmailmodeValue, sendmailmodeLabel) in sendmailModes"
:value="sendmailmodeValue"
:selected="sendmailmodeValue === 'mailSendmailMode'">
{{ sendmailModes[sendmailmodeLabel] }}
</option>
</select>
</div>

<p class="form-control">
<label for="mail_from_address">{{ t('settings', 'From address') }}</label>
<input type="text" name="mail_from_address" id="mail_from_address" placeholder="noreply"
v-model="mailFromAddress" />
@
<input type="text" name="mail_domain" id="mail_domain" placeholder="example.com"
v-model="mailDomain" />
</p>

<p v-if="mailSmtpMode === 'smtp'" class="form-control">
<label for="mail_smtpauthtype">{{ t('settings', 'Authentication method') }}</label>
<select name="mail_smtpauthtype" id="mail_smtpauthtype">
<option v-for="(name, authType) in smtpAuthType"
:key="authType"
:value="authType"
:selected="authType === mailSmtpAuthType">
{{ name }}
</option>
</select>

<NcCheckboxRadioSwitch type="checkbox"
:checked.sync="mailSmtpAuth">
{{ t('settings', 'Authentication required') }}
</NcCheckboxRadioSwitch>
</p>

<p v-if="mailSmtpMode === 'smtp'" class="form-control">
<label for="mail_smtphost">{{ t('settings', 'Server address') }}</label>
<input type="text"
name="mail_smtphost"
id="mail_smtphost"
placeholder="smtp.example.com"
v-model="mailSmtpHost" />
:
<input type="text"
inputmode="numeric"
name="mail_smtpport"
id="mail_smtpport"
:placeholder="t('settings', 'Port')"
v-model="mailSmtpPort" />
</p>
</div>

<form v-if="mailSmtpAuth && mailSmtpMode === 'smtp'" class="topMargin">
<h3>{{ t('settings', 'Authentication') }}</h3>
<p>
<label for="mail_smtpname">{{ t('settings', 'SMTP username') }}</label>
<input type="text"
name="mail_smtpname"
id="mail_smtpname"
:value="mailSmtpName" />
</p>
<p>
<label for="mailSmtpPassword">{{ t('settings', 'SMTP password') }}</label>
<input type="text"
id="mail_smtppassword"
autocomplete="off"
:value="mailSmtpPassword" />
<NcButton nativeType="submit">
{{ t('settings', 'Save') }}
</NcButton>
</p>
</form>

<div class="topMargin">
<p>{{ t('settings', 'Test and verify email settings') }}</p>
<NcButton @click="sendMail">
{{ t('settings', 'Send email') }}
</NcButton>
<p v-id="sendMailMessage">{{ sendMailMessage }}</p>
</div>
</NcSettingsSection>
</template>

<script>

import axios from '@nextcloud/axios'
import NcCheckboxRadioSwitch from '@nextcloud/vue/dist/Components/NcCheckboxRadioSwitch'
import NcButton from '@nextcloud/vue/dist/Components/NcButton'
import NcSettingsSection from '@nextcloud/vue/dist/Components/NcSettingsSection'
import { loadState } from '@nextcloud/initial-state'
import { getLoggerBuilder } from '@nextcloud/logger'
import { generateOcsUrl } from '@nextcloud/router'
import { confirmPassword } from '@nextcloud/password-confirmation'
import '@nextcloud/password-confirmation/dist/style.css'
import { showError } from '@nextcloud/dialogs'

const smtpAuthType = {
'': t('settings', 'None'),
'LOGIN': t('settings', 'Login'),
'PLAIN': t('settings', 'Plain'),
'NTLM': t('settings', 'NT LAN Manager'),
}

const smtpSecure = {
'': t('settings', 'None'),
'ssl': t('settings', 'SSL/TLS'),
'tls': t('settings', 'STARTTLS'),
}

const smtpModes = [
['smtp', 'SMTP'],
]

if (loadState('settings', 'sendmail_is_available')) {
smtpModes.push(['sendmail', 'Sendmail'])
}
if (loadState('settings', 'mail_smtpmode') === 'qmail') {
// legacy only show it if it was previously enabled
smtpModes.push(['qmail', 'qmail'])
}

const sendmailModes = {
'smtp': 'smtp (-bs)',
'pipe': 'pipe (-t)'
}

const logger = getLoggerBuilder()
.setApp('settings')
.detectUser()
.build()

export default {
name: 'MailDeliverySettings',
components: {
NcCheckboxRadioSwitch,
NcSettingsSection,
NcButton,
},
data() {
return {
smtpAuthType,
smtpSecure,
smtpModes,
sendmailModes,

mailFromAddress: loadState('settings', 'mail_from_address'),
mailDomain: loadState('settings', 'mail_domain'),
mailSmtpMode: loadState('settings', 'mail_smtpmode'),
mailSmtpSecure: loadState('settings', 'mail_smtpsecure'),
mailSmtpHost: loadState('settings', 'mail_smtphost'),
mailSmtpPort: loadState('settings', 'mail_smtpport'),
mailSmtpAuthType: loadState('settings', 'mail_smtpauthtype'),
mailSmtpAuth: loadState('settings', 'mail_smtpauth'),
mailSmtpName: loadState('settings', 'mail_smtpname'),
mailSmtpPassword: loadState('settings', 'mail_smtppassword'),
mailSendmailMode: loadState('settings', 'mail_sendmailmode'),
adminDocUrl: loadState('settings', 'emailAdminDocUrl'),
}
},
methods: {
async update(key, value) {
await confirmPassword()

const url = generateOcsUrl('/apps/provisioning_api/api/v1/config/apps/{appId}/{key}', {
appId: 'core',
key,
})

const stringValue = value ? 'yes' : 'no'
try {
const { data } = await axios.post(url, {
value: stringValue,
})
this.handleResponse({
status: data.ocs?.meta?.status,
})
} catch (e) {
this.handleResponse({
errorMessage: t('settings', 'Unable to update server side encryption config'),
error: e,
})
}
},
async handleResponse({ status, errorMessage, error }) {
if (status !== 'ok') {
showError(errorMessage)
logger.error(errorMessage, { error })
}
},
},
}
</script>

<style lang="scss" scopped>
.topMargin {
margin-top: 2rem;
}
.form-control {
label {
display: block;
padding: 4px 0;
}
}
</style>
35 changes: 35 additions & 0 deletions apps/settings/src/main-admin-email.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
/**
* @copyright 2022 Carl Schwa <[email protected]>
* @license AGPL-3.0-or-later
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/

import { loadState } from '@nextcloud/initial-state'
import Vue from 'vue'

import MailDeliverySettings from './components/MailDeliverySettings.vue'

// eslint-disable-next-line camelcase
__webpack_nonce__ = btoa(OC.requestToken)

Vue.prototype.t = t

// Not used here but required for legacy templates
window.OC = window.OC || {}
window.OC.Settings = window.OC.Settings || {}

const View = Vue.extend(MailDeliverySettings)
new View().$mount('#mail_delivery_settings')
Loading