aboutsummaryrefslogtreecommitdiffstats
path: root/apps/settings/src
diff options
context:
space:
mode:
Diffstat (limited to 'apps/settings/src')
-rw-r--r--apps/settings/src/admin.js335
-rw-r--r--apps/settings/src/apps.js6
-rw-r--r--apps/settings/src/mixins/AppManagement.js27
-rw-r--r--apps/settings/src/router.js2
-rw-r--r--apps/settings/src/service/rebuild-navigation.js130
-rw-r--r--apps/settings/src/store/apps.js16
6 files changed, 496 insertions, 20 deletions
diff --git a/apps/settings/src/admin.js b/apps/settings/src/admin.js
new file mode 100644
index 00000000000..24f95ccb2a6
--- /dev/null
+++ b/apps/settings/src/admin.js
@@ -0,0 +1,335 @@
+import $ from 'jquery'
+import 'jquery-ui-dist/jquery-ui'
+
+window.addEventListener('DOMContentLoaded', () => {
+ $('#excludedGroups,#linksExcludedGroups').each((index, element) => {
+ OC.Settings.setupGroupsSelect($(element))
+ $(element).change((ev) => {
+ let groups = ev.val || []
+ groups = JSON.stringify(groups)
+ OCP.AppConfig.setValue('core', $(this).attr('name'), groups)
+ })
+ })
+
+ $('#loglevel').change(() => {
+ $.post(OC.generateUrl('/settings/admin/log/level'), { level: $(this).val() }, () => {
+ OC.Log.reload()
+ })
+ })
+
+ $('#backgroundjobs span.crondate').tooltip({ placement: 'top' })
+
+ $('#backgroundjobs input').change(() => {
+ if ($(this).is(':checked')) {
+ const mode = $(this).val()
+ if (mode === 'ajax' || mode === 'webcron' || mode === 'cron') {
+ OCP.AppConfig.setValue('core', 'backgroundjobs_mode', mode, {
+ success: () => {
+ // clear cron errors on background job mode change
+ OCP.AppConfig.deleteKey('core', 'cronErrors')
+ }
+ })
+ }
+ }
+ })
+
+ $('#shareAPIEnabled').change(() => {
+ $('#shareAPI p:not(#enable)').toggleClass('hidden', !this.checked)
+ })
+
+ $('#enableEncryption').change(() => {
+ $('#encryptionAPI div#EncryptionWarning').toggleClass('hidden')
+ })
+
+ $('#reallyEnableEncryption').click(() => {
+ $('#encryptionAPI div#EncryptionWarning').toggleClass('hidden')
+ $('#encryptionAPI div#EncryptionSettingsArea').toggleClass('hidden')
+ OCP.AppConfig.setValue('core', 'encryption_enabled', 'yes')
+ $('#enableEncryption').attr('disabled', 'disabled')
+ })
+
+ $('#startmigration').click((event) => {
+ $(window).on('beforeunload.encryption', (e) => {
+ return t('settings', 'Migration in progress. Please wait until the migration is finished')
+ })
+ event.preventDefault()
+ $('#startmigration').prop('disabled', true)
+ OC.msg.startAction('#startmigration_msg', t('settings', 'Migration started …'))
+ $.post(OC.generateUrl('/settings/admin/startmigration'), '', function(data) {
+ OC.msg.finishedAction('#startmigration_msg', data)
+ if (data.status === 'success') {
+ $('#encryptionAPI div#selectEncryptionModules').toggleClass('hidden')
+ $('#encryptionAPI div#migrationWarning').toggleClass('hidden')
+ } else {
+ $('#startmigration').prop('disabled', false)
+ }
+ $(window).off('beforeunload.encryption')
+
+ })
+ })
+
+ $('#shareapiExpireAfterNDays').on('input', function() {
+ this.value = this.value.replace(/\D/g, '')
+ })
+
+ $('#shareAPI input:not(.noJSAutoUpdate)').change(function() {
+ let value = $(this).val()
+ if ($(this).attr('type') === 'checkbox') {
+ if (this.checked) {
+ value = 'yes'
+ } else {
+ value = 'no'
+ }
+ }
+ OCP.AppConfig.setValue('core', $(this).attr('name'), value)
+ })
+
+ $('#shareapiDefaultExpireDate').change(function() {
+ $('setDefaultExpireDate').toggleClass('hidden', !this.checked)
+ })
+
+ $('#shareapiDefaultInternalExpireDate').change(function() {
+ $('#setDefaultInternalExpireDate').toggleClass('hidden', !this.checked)
+ })
+
+ $('#shareapiDefaultRemoteExpireDate').change(function() {
+ $('#setDefaultRemoteExpireDate').toggleClass('hidden', !this.checked)
+ })
+
+ $('#publicShareDisclaimer').change(function() {
+ $('#publicShareDisclaimerText').toggleClass('hidden', !this.checked)
+ if (!this.checked) {
+ savePublicShareDisclaimerText('')
+ }
+ })
+
+ $('#shareApiDefaultPermissionsSection input').change(function(ev) {
+ const $el = $('#shareApiDefaultPermissions')
+ const $target = $(ev.target)
+
+ let value = $el.val()
+ if ($target.is(':checked')) {
+ value = value | $target.val()
+ } else {
+ value = value & ~$target.val()
+ }
+
+ // always set read permission
+ value |= OC.PERMISSION_READ
+
+ // this will trigger the field's change event and will save it
+ $el.val(value).change()
+
+ ev.preventDefault()
+
+ return false
+ })
+
+ const savePublicShareDisclaimerText = _.debounce(function(value) {
+ const options = {
+ success: () => {
+ OC.msg.finishedSuccess('#publicShareDisclaimerStatus', t('settings', 'Saved'))
+ },
+ error: () => {
+ OC.msg.finishedError('#publicShareDisclaimerStatus', t('settings', 'Not saved'))
+ }
+ }
+
+ OC.msg.startSaving('#publicShareDisclaimerStatus')
+ if (_.isString(value) && value !== '') {
+ OCP.AppConfig.setValue('core', 'shareapi_public_link_disclaimertext', value, options)
+ } else {
+ $('#publicShareDisclaimerText').val('')
+ OCP.AppConfig.deleteKey('core', 'shareapi_public_link_disclaimertext', options)
+ }
+ }, 500)
+
+ $('#publicShareDisclaimerText').on('change, keyup', function() {
+ savePublicShareDisclaimerText(this.value)
+ })
+
+ $('#shareapi_allow_share_dialog_user_enumeration').on('change', function() {
+ $('#shareapi_restrict_user_enumeration_to_group_setting').toggleClass('hidden', !this.checked)
+ $('#shareapi_restrict_user_enumeration_to_phone_setting').toggleClass('hidden', !this.checked)
+ $('#shareapi_restrict_user_enumeration_combinewarning_setting').toggleClass('hidden', !this.checked)
+ })
+
+ $('#allowLinks').change(function() {
+ $('#publicLinkSettings').toggleClass('hidden', !this.checked)
+ $('#setDefaultExpireDate').toggleClass('hidden', !(this.checked && $('#shareapiDefaultExpireDate')[0].checked))
+ })
+
+ $('#mail_smtpauth').change(function() {
+ if (!this.checked) {
+ $('#mail_credentials').addClass('hidden')
+ } else {
+ $('#mail_credentials').removeClass('hidden')
+ }
+ })
+
+ $('#mail_smtpmode').change(function() {
+ if ($(this).val() !== 'smtp') {
+ $('#setting_smtpauth').addClass('hidden')
+ $('#setting_smtphost').addClass('hidden')
+ $('#mail_smtpsecure_label').addClass('hidden')
+ $('#mail_smtpsecure').addClass('hidden')
+ $('#mail_credentials').addClass('hidden')
+ $('#mail_sendmailmode_label, #mail_sendmailmode').removeClass('hidden')
+ } else {
+ $('#setting_smtpauth').removeClass('hidden')
+ $('#setting_smtphost').removeClass('hidden')
+ $('#mail_smtpsecure_label').removeClass('hidden')
+ $('#mail_smtpsecure').removeClass('hidden')
+ if ($('#mail_smtpauth').is(':checked')) {
+ $('#mail_credentials').removeClass('hidden')
+ }
+ $('#mail_sendmailmode_label, #mail_sendmailmode').addClass('hidden')
+ }
+ })
+
+ const changeEmailSettings = function() {
+ if (OC.PasswordConfirmation.requiresPasswordConfirmation()) {
+ OC.PasswordConfirmation.requirePasswordConfirmation(changeEmailSettings)
+ return
+ }
+
+ OC.msg.startSaving('#mail_settings_msg')
+ $.ajax({
+ url: OC.generateUrl('/settings/admin/mailsettings'),
+ type: 'POST',
+ data: $('#mail_general_settings_form').serialize(),
+ success: () => {
+ OC.msg.finishedSuccess('#mail_settings_msg', t('settings', 'Saved'))
+ },
+ error: (xhr) => {
+ OC.msg.finishedError('#mail_settings_msg', xhr.responseJSON)
+ }
+ })
+ }
+
+ const toggleEmailCredentials = function() {
+ if (OC.PasswordConfirmation.requiresPasswordConfirmation()) {
+ OC.PasswordConfirmation.requirePasswordConfirmation(toggleEmailCredentials)
+ return
+ }
+
+ OC.msg.startSaving('#mail_settings_msg')
+ $.ajax({
+ url: OC.generateUrl('/settings/admin/mailsettings/credentials'),
+ type: 'POST',
+ data: $('#mail_credentials_settings').serialize(),
+ success: () => {
+ OC.msg.finishedSuccess('#mail_settings_msg', t('settings', 'Saved'))
+ },
+ error: (xhr) => {
+ OC.msg.finishedError('#mail_settings_msg', xhr.responseJSON)
+ }
+ })
+ }
+
+ $('#mail_general_settings_form').change(changeEmailSettings)
+ $('#mail_credentials_settings_submit').click(toggleEmailCredentials)
+ $('#mail_smtppassword').click(() => {
+ if (this.type === 'text' && this.value === '********') {
+ this.type = 'password'
+ this.value = ''
+ }
+ })
+
+ $('#sendtestemail').click((event) => {
+ event.preventDefault()
+ OC.msg.startAction('#sendtestmail_msg', t('settings', 'Sending…'))
+
+ $.ajax({
+ url: OC.generateUrl('/settings/admin/mailtest'),
+ type: 'POST',
+ success: () => {
+ OC.msg.finishedSuccess('#sendtestmail_msg', t('settings', 'Email sent'))
+ },
+ error: (xhr) => {
+ OC.msg.finishedError('#sendtestmail_msg', xhr.responseJSON)
+ }
+ })
+ })
+
+ $('#allowGroupSharing').change(() => {
+ $('#allowGroupSharing').toggleClass('hidden', !this.checked)
+ })
+
+ $('#shareapiExcludeGroups').change(() => {
+ $('#selectExcludedGroups').toggleClass('hidden', !this.checked)
+ })
+
+ const setupChecks = () => {
+ // run setup checks then gather error messages
+ $.when(
+ OC.SetupChecks.checkWebDAV(),
+ OC.SetupChecks.checkWellKnownUrl('GET', '/.well-known/webfinger', OC.theme.docPlaceholderUrl, $('#postsetupchecks').data('check-wellknown') === true, [200, 404], true),
+ OC.SetupChecks.checkWellKnownUrl('GET', '/.well-known/nodeinfo', OC.theme.docPlaceholderUrl, $('#postsetupchecks').data('check-wellknown') === true, [200, 404], true),
+ OC.SetupChecks.checkWellKnownUrl('PROPFIND', '/.well-known/caldav', OC.theme.docPlaceholderUrl, $('#postsetupchecks').data('check-wellknown') === true),
+ OC.SetupChecks.checkWellKnownUrl('PROPFIND', '/.well-known/carddav', OC.theme.docPlaceholderUrl, $('#postsetupchecks').data('check-wellknown') === true),
+ OC.SetupChecks.checkProviderUrl(OC.getRootPath() + '/ocm-provider/', OC.theme.docPlaceholderUrl, $('#postsetupchecks').data('check-wellknown') === true),
+ OC.SetupChecks.checkProviderUrl(OC.getRootPath() + '/ocs-provider/', OC.theme.docPlaceholderUrl, $('#postsetupchecks').data('check-wellknown') === true),
+ OC.SetupChecks.checkSetup(),
+ OC.SetupChecks.checkGeneric(),
+ OC.SetupChecks.checkWOFF2Loading(OC.filePath('core', '', 'fonts/NotoSans-Regular-latin.woff2'), OC.theme.docPlaceholderUrl),
+ OC.SetupChecks.checkDataProtected()
+ ).then((check1, check2, check3, check4, check5, check6, check7, check8, check9, check10, check11) => {
+ const messages = [].concat(check1, check2, check3, check4, check5, check6, check7, check8, check9, check10, check11)
+ const $el = $('#postsetupchecks')
+ $('#security-warning-state-loading').addClass('hidden')
+
+ let hasMessages = false
+ const $errorsEl = $el.find('.errors')
+ const $warningsEl = $el.find('.warnings')
+ const $infoEl = $el.find('.info')
+
+ for (let i = 0; i < messages.length; i++) {
+ switch (messages[i].type) {
+ case OC.SetupChecks.MESSAGE_TYPE_INFO:
+ $infoEl.append('<li>' + messages[i].msg + '</li>')
+ break
+ case OC.SetupChecks.MESSAGE_TYPE_WARNING:
+ $warningsEl.append('<li>' + messages[i].msg + '</li>')
+ break
+ case OC.SetupChecks.MESSAGE_TYPE_ERROR:
+ default:
+ $errorsEl.append('<li>' + messages[i].msg + '</li>')
+ }
+ }
+
+ if ($errorsEl.find('li').length > 0) {
+ $errorsEl.removeClass('hidden')
+ hasMessages = true
+ }
+ if ($warningsEl.find('li').length > 0) {
+ $warningsEl.removeClass('hidden')
+ hasMessages = true
+ }
+ if ($infoEl.find('li').length > 0) {
+ $infoEl.removeClass('hidden')
+ hasMessages = true
+ }
+
+ if (hasMessages) {
+ $('#postsetupchecks-hint').removeClass('hidden')
+ if ($errorsEl.find('li').length > 0) {
+ $('#security-warning-state-failure').removeClass('hidden')
+ } else {
+ $('#security-warning-state-warning').removeClass('hidden')
+ }
+ } else {
+ const securityWarning = $('#security-warning')
+ if (securityWarning.children('ul').children().length === 0) {
+ $('#security-warning-state-ok').removeClass('hidden')
+ } else {
+ $('#security-warning-state-failure').removeClass('hidden')
+ }
+ }
+ })
+ }
+
+ if (document.getElementById('security-warning') !== null) {
+ setupChecks()
+ }
+})
diff --git a/apps/settings/src/apps.js b/apps/settings/src/apps.js
new file mode 100644
index 00000000000..fafdb531399
--- /dev/null
+++ b/apps/settings/src/apps.js
@@ -0,0 +1,6 @@
+import rebuildNavigation from './service/rebuild-navigation.js'
+
+window.OC.Settings = window.OC.Settings || {}
+window.OC.Settings.Apps = window.OC.Settings.Apps || {
+ rebuildNavigation
+}
diff --git a/apps/settings/src/mixins/AppManagement.js b/apps/settings/src/mixins/AppManagement.js
index 96f2c1dad4a..cf44a37a53f 100644
--- a/apps/settings/src/mixins/AppManagement.js
+++ b/apps/settings/src/mixins/AppManagement.js
@@ -20,6 +20,9 @@
*
*/
+import { showError } from '@nextcloud/dialogs'
+import rebuildNavigation from '../service/rebuild-navigation.js'
+
export default {
computed: {
appGroups() {
@@ -109,33 +112,33 @@ export default {
},
forceEnable(appId) {
this.$store.dispatch('forceEnableApp', { appId, groups: [] })
- .then((response) => { OC.Settings.Apps.rebuildNavigation() })
- .catch((error) => { OC.Notification.show(error) })
+ .then((response) => { rebuildNavigation() })
+ .catch((error) => { showError(error) })
},
enable(appId) {
this.$store.dispatch('enableApp', { appId, groups: [] })
- .then((response) => { OC.Settings.Apps.rebuildNavigation() })
- .catch((error) => { OC.Notification.show(error) })
+ .then((response) => { rebuildNavigation() })
+ .catch((error) => { showError(error) })
},
disable(appId) {
this.$store.dispatch('disableApp', { appId })
- .then((response) => { OC.Settings.Apps.rebuildNavigation() })
- .catch((error) => { OC.Notification.show(error) })
+ .then((response) => { rebuildNavigation() })
+ .catch((error) => { showError(error) })
},
remove(appId) {
this.$store.dispatch('uninstallApp', { appId })
- .then((response) => { OC.Settings.Apps.rebuildNavigation() })
- .catch((error) => { OC.Notification.show(error) })
+ .then((response) => { rebuildNavigation() })
+ .catch((error) => { showError(error) })
},
install(appId) {
this.$store.dispatch('enableApp', { appId })
- .then((response) => { OC.Settings.Apps.rebuildNavigation() })
- .catch((error) => { OC.Notification.show(error) })
+ .then((response) => { rebuildNavigation() })
+ .catch((error) => { showError(error) })
},
update(appId) {
this.$store.dispatch('updateApp', { appId })
- .then((response) => { OC.Settings.Apps.rebuildNavigation() })
- .catch((error) => { OC.Notification.show(error) })
+ .then((response) => { rebuildNavigation() })
+ .catch((error) => { showError(error) })
},
},
}
diff --git a/apps/settings/src/router.js b/apps/settings/src/router.js
index 202702a94df..912e031c0d7 100644
--- a/apps/settings/src/router.js
+++ b/apps/settings/src/router.js
@@ -28,7 +28,7 @@ import { generateUrl } from '@nextcloud/router'
// Dynamic loading
const Users = () => import(/* webpackChunkName: 'settings-users' */'./views/Users')
-const Apps = () => import(/* webpackChunkName: 'settings-apps' */'./views/Apps')
+const Apps = () => import(/* webpackChunkName: 'settings-apps-view' */'./views/Apps')
Vue.use(Router)
diff --git a/apps/settings/src/service/rebuild-navigation.js b/apps/settings/src/service/rebuild-navigation.js
new file mode 100644
index 00000000000..57cb1e439bd
--- /dev/null
+++ b/apps/settings/src/service/rebuild-navigation.js
@@ -0,0 +1,130 @@
+import axios from '@nextcloud/axios'
+import { generateOcsUrl } from '@nextcloud/router'
+
+export default () => {
+ return axios.get(generateOcsUrl('core/navigation', 2) + '/apps?format=json')
+ .then(({ data }) => {
+ if (data.ocs.meta.statuscode !== 200) {
+ return
+ }
+
+ const addedApps = {}
+ const navEntries = data.ocs.data
+ const container = document.querySelector('#navigation #apps ul')
+
+ // remove disabled apps
+ navEntries.forEach((entry) => {
+ if (!container.querySelector('li[data-id="' + entry.id + '"]')) {
+ addedApps[entry.id] = true
+ }
+ })
+
+ container.querySelectorAll('li[data-id]').forEach((el, index) => {
+ const id = el.dataset.id
+ // remove all apps that are not in the correct order
+ if (!navEntries[index] || (navEntries[index] && navEntries[index].id !== id)) {
+ el.remove()
+ document.querySelector(`#appmenu li[data-id=${id}]`).remove()
+ }
+ })
+
+ let previousEntry = {}
+ // add enabled apps to #navigation and #appmenu
+ navEntries.forEach((entry) => {
+ if (container.querySelector(`li[data-id="${entry.id}"]`) === null) {
+ const li = document.createElement('li')
+ li.dataset.id = entry.id
+ const img = `<svg width="20" height="20" viewBox="0 0 20 20" alt="">
+ <defs>
+ <filter id="invertMenuMore-${entry.id}"><feColorMatrix in="SourceGraphic" type="matrix" values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0"></feColorMatrix></filter>
+ <mask id="hole">
+ <rect width="100%" height="100%" fill="white"></rect>
+ <circle r="4.5" cx="17" cy="3" fill="black"></circle>
+ </mask>
+ </defs>
+ <image x="0" y="0" width="16" height="16" filter="url(#invertMenuMore-${entry.id})" preserveAspectRatio="xMinYMin meet" xlink:href="${entry.icon}" class="app-icon" />
+ </svg>`
+
+ const imgElement = document.createElement('template')
+ imgElement.innerHTML = img
+
+ const a = document.createElement('a')
+ a.setAttribute('href', entry.href)
+
+ const filename = document.createElement('span')
+ filename.appendChild(document.createTextNode(entry.name))
+
+ const loading = document.createElement('div')
+ loading.setAttribute('class', 'unread-counter')
+ loading.style.display = 'none'
+
+ // draw attention to the newly added app entry
+ // by flashing twice the more apps menu
+ if (addedApps[entry.id]) {
+ a.classList.add('animated')
+ }
+
+ a.prepend(imgElement.content.firstChild, loading, filename)
+ li.append(a)
+
+ // add app icon to the navigation
+ const previousElement = document.querySelector(`#navigation li[data-id=${previousEntry.id}]`)
+ if (previousElement) {
+ previousElement.insertAdjacentElement('afterend', li)
+ } else {
+ document.querySelector('#navigation #apps ul').prepend(li)
+ }
+ }
+
+ if (document.getElementById('appmenu').querySelector(`li[data-id="${entry.id}"]`) === null) {
+ const li = document.createElement('li')
+ li.dataset.id = entry.id
+ // Generating svg embedded image (see layout.user.php)
+ let img
+ if (OCA.Theming && OCA.Theming.inverted) {
+ img = `<svg width="20" height="20" viewBox="0 0 20 20" alt="">
+ <defs>
+ <filter id="invert"><feColorMatrix in="SourceGraphic" type="matrix" values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" /></filter>
+ </defs>
+ <image x="0" y="0" width="20" height="20" preserveAspectRatio="xMinYMin meet" filter="url(#invert)" xlink:href="${entry.icon}" class="app-icon" />
+ </svg>`
+ } else {
+ img = `<svg width="20" height="20" viewBox="0 0 20 20" alt="">
+ <image x="0" y="0" width="20" height="20" preserveAspectRatio="xMinYMin meet" xlink:href="${entry.icon}" class="app-icon" />
+ </svg>`
+ }
+ const imgElement = document.createElement('template')
+ imgElement.innerHTML = img
+
+ const a = document.createElement('a')
+ a.setAttribute('href', entry.href)
+
+ const filename = document.createElement('span')
+ filename.appendChild(document.createTextNode(entry.name))
+
+ const loading = document.createElement('div')
+ loading.setAttribute('class', 'icon-loading-dark')
+ loading.style.display = 'none'
+
+ // draw attention to the newly added app entry
+ // by flashing twice the more apps menu
+ if (addedApps[entry.id]) {
+ a.classList.add('animated')
+ }
+
+ a.prepend(loading, filename, imgElement.content.firstChild)
+ li.append(a)
+
+ // add app icon to the navigation
+ const previousElement = document.querySelector('#appmenu li[data-id=' + previousEntry.id + ']')
+ if (previousElement) {
+ previousElement.insertAdjacentElement('afterend', li)
+ } else {
+ document.queryElementById('appmenu').prepend(li)
+ }
+ }
+ previousEntry = entry
+ })
+ window.dispatchEvent(new Event('resize'))
+ })
+}
diff --git a/apps/settings/src/store/apps.js b/apps/settings/src/store/apps.js
index 038b42e64c9..1efed274b32 100644
--- a/apps/settings/src/store/apps.js
+++ b/apps/settings/src/store/apps.js
@@ -25,6 +25,8 @@
import api from './api'
import Vue from 'vue'
import { generateUrl } from '@nextcloud/router'
+import { showError, showInfo } from '@nextcloud/dialogs'
+import '@nextcloud/dialogs/styles/toast.scss'
const state = {
apps: [],
@@ -37,7 +39,7 @@ const state = {
const mutations = {
APPS_API_FAILURE(state, error) {
- OC.Notification.showHtml(t('settings', 'An error occured during the request. Unable to proceed.') + '<br>' + error.error.response.data.data.message, { timeout: 7 })
+ showError(t('settings', 'An error occured during the request. Unable to proceed.') + '<br>' + error.error.response.data.data.message, { timeout: 7, isHTML: true })
console.error(state, error)
},
@@ -180,16 +182,16 @@ const actions = {
return api.get(generateUrl('apps/files'))
.then(() => {
if (response.data.update_required) {
- OC.dialogs.info(
+ showInfo(
t(
'settings',
'The app has been enabled but needs to be updated. You will be redirected to the update page in 5 seconds.'
),
- t('settings', 'App update'),
- function() {
- window.location.reload()
- },
- true
+ {
+ onClick: () => window.location.reload(),
+ close: false,
+
+ }
)
setTimeout(function() {
location.reload()