1
0
Fork 0
mirror of https://gitnet.fr/deblan/side_menu.git synced 2025-12-17 21:02:25 +01:00
side_menu/src/pages/AdminSettings.vue
2025-04-16 09:13:12 +02:00

611 lines
17 KiB
Vue

<!--
@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>
<NcContent
v-if="config"
app-name="side_menu"
>
<NcAppContent>
<div class="side-menu-setting">
<NcButton
v-for="item in menu"
:key="item.label"
:variant="item.section === section ? 'primary' : 'secondary'"
@click="setSection(item.section)"
>{{ trans(item.label) }}</NcButton
>
</div>
<TableContainer :class="sectionClass('panel')">
<TableRow>
<TableLabel label="Display" />
<TableValue>
<FormDisplayPicker
:always-displayed="config['always-displayed']"
:top-wide-menu="config['big-menu']"
:side-menu-with-categories="config['side-with-categories']"
@update:always-displayed="(value) => (config['always-displayed'] = value)"
@update:top-wide-menu="(value) => (config['big-menu'] = value)"
@update:side-menu-with-categories="(value) => (config['side-with-categories'] = value)"
/>
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Display the logo"
:middle="true"
/>
<TableValue>
<FormYesNo v-model="config['display-logo']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Use the avatar instead of the logo"
:middle="true"
/>
<TableValue>
<FormYesNo v-model="config['use-avatar']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="The logo is a link to the default app"
:middle="true"
/>
<TableValue>
<FormYesNo v-model="config['add-logo-link']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Show the link to settings"
:middle="true"
/>
<TableValue>
<FormYesNo v-model="config['show-settings']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Icons"
:middle="true"
/>
<TableValue>
<FormSize v-model="config['size-icon']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Texts"
:middle="true"
/>
<TableValue>
<FormSize v-model="config['size-text']" />
</TableValue>
</TableRow>
</TableContainer>
<TableContainer :class="sectionClass('topMenu')">
<TableRow>
<TableLabel
label="Applications kept in the top menu"
:middle="true"
/>
<TableValue>
<FormAppPicker v-model="config['top-menu-apps']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Applications kept in the top menu but also shown in side menu"
help="These applications must be selected in the previous option."
:middle="true"
/>
<TableValue>
<FormAppPicker v-model="config['top-side-menu-apps']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Hide labels on mouse over"
:top="true"
/>
<TableValue>
<FormSelect
v-model="config['top-menu-mouse-over-hidden-label']"
:expanded="true"
:options="[
{ id: 1, label: 'Yes' },
{ id: 0, label: 'No' },
{ id: 2, label: 'Except the hovered app' },
]"
/>
</TableValue>
</TableRow>
</TableContainer>
<TableContainer :class="sectionClass('apps')">
<TableRow>
<TableLabel
label="Apps that should not be displayed in the menu"
:middle="true"
/>
<TableValue>
<FormAppPicker v-model="config['big-menu-hidden-apps']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Open apps in new tab"
:middle="true"
/>
<TableValue>
<FormAppPicker v-model="config['target-blank-apps']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Customize sorting"
:middle="true"
/>
<TableValue>
<FormAppSort v-model="config['apps-order']" />
</TableValue>
</TableRow>
</TableContainer>
<TableContainer :class="sectionClass('opener')">
<TableRow>
<TableLabel
label="Opener"
:middle="true"
/>
<TableValue>
<FormOpener v-model="config['opener']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Dark mode opener"
:middle="true"
/>
<TableValue>
<FormOpener v-model="config['dark-mode-opener']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Position"
:middle="true"
/>
<TableValue>
<FormSelect
v-model="config['opener-position']"
:options="[
{ id: 'before', label: 'Before the logo' },
{ id: 'after', label: 'After the logo' },
]"
/>
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Show only the opener (hidden logo)"
:middle="true"
/>
<TableValue>
<FormYesNo v-model="config['opener-only']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Open the menu when the mouse is hover the opener (automatically disabled on touch screens)"
help="This is the automatic behavior when the menu is always displayed."
:middle="true"
/>
<TableValue>
<FormYesNo v-model="config['opener-hover']" />
</TableValue>
</TableRow>
</TableContainer>
<TableContainer :class="sectionClass('colors')">
<SectionTitle label="Colors" />
<TableRow>
<TableLabel
label="Background color"
:middle="true"
/>
<TableValue>
<FormColorPicker v-model="config['background-color']" />
<FormColorPicker v-model="config['background-color-to']" />
<FormRange
v-model="config['background-color-opacity']"
prepend="Transparent"
append="Opaque"
/>
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Background color of current app"
:middle="true"
/>
<TableValue>
<FormColorPicker v-model="config['current-app-background-color']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Text color"
:middle="true"
/>
<TableValue>
<FormColorPicker v-model="config['text-color']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Loader"
:middle="true"
/>
<TableValue>
<FormColorPicker v-model="config['loader-color']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Icon"
:middle="true"
/>
<TableValue>
<FormRange
v-model="config['icon-invert-filter']"
prepend="Same color"
append="Opposite color"
/>
<FormRange
v-model="config['icon-opacity']"
prepend="Transparent"
append="Opaque"
/>
</TableValue>
</TableRow>
<SectionTitle label="Dark mode colors" />
<TableRow>
<TableLabel
label="Background color"
:middle="true"
/>
<TableValue>
<FormColorPicker v-model="config['dark-mode-background-color']" />
<FormColorPicker v-model="config['dark-mode-background-color-to']" />
<FormRange
v-model="config['dark-mode-background-color-opacity']"
prepend="Transparent"
append="Opaque"
/>
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Background color of current app"
:middle="true"
/>
<TableValue>
<FormColorPicker v-model="config['dark-mode-current-app-background-color']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Text color"
:middle="true"
/>
<TableValue>
<FormColorPicker v-model="config['dark-mode-text-color']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Loader"
:middle="true"
/>
<TableValue>
<FormColorPicker v-model="config['dark-mode-loader-color']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Icon"
:middle="true"
/>
<TableValue>
<FormRange
v-model="config['dark-mode-icon-invert-filter']"
prepend="Same color"
append="Opposite color"
/>
<FormRange
v-model="config['dark-mode-icon-opacity']"
prepend="Transparent"
append="Opaque"
/>
</TableValue>
</TableRow>
</TableContainer>
<TableContainer :class="sectionClass('global')">
<TableRow>
<TableLabel
label="The menu is enabled by default for users"
help="Except when the configuration is forced."
:middle="true"
/>
<TableValue>
<FormYesNo v-model="config['default-enabled']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Force this configuration to users"
:middle="true"
/>
<TableValue>
<FormYesNo v-model="config['force']" />
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Loader enabled"
:middle="true"
/>
<TableValue>
<FormYesNo v-model="config['loader-enabled']" />
</TableValue>
</TableRow>
</TableContainer>
<TableContainer :class="sectionClass('support')">
<TableRow>
<TableLabel
label="You like this app and you want to support me?"
:middle="true"
/>
<TableValue>
<a
target="_blank"
href="https://www.buymeacoffee.com/deblan"
rel="noopener"
>
<NcButton variant="secondary">{{ trans('Buy me a coffee ☕') }}</NcButton>
</a>
</TableValue>
</TableRow>
<TableRow>
<TableLabel label="Need help" />
<TableValue class="button-inline">
<a
target="_blank"
href="https://deblan.gitnet.page/side_menu_doc/"
rel="noopener"
>
<NcButton variant="secondary">{{ trans('Open the documentation') }}</NcButton>
</a>
<a
target="_blank"
href="https://gitnet.fr/deblan/side_menu/issues/new?template=.gitea%2fissue_template%2fQUESTION_TEMPLATE.yml"
rel="noopener"
>
<NcButton variant="secondary">{{ trans('Ask the developer') }}</NcButton>
</a>
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="I would like a new feature"
:middle="true"
/>
<TableValue>
<a
target="_blank"
href="https://gitnet.fr/deblan/side_menu/issues/new?template=.gitea%2fissue_template%2fFEATURE_TEMPLATE.yml"
rel="noopener"
>
<NcButton variant="secondary">{{ trans('New request') }}</NcButton>
</a>
</TableValue>
</TableRow>
<TableRow>
<TableLabel
label="Something went wrong"
:top="true"
/>
<TableValue class="button-inline">
<a
target="_blank"
href="https://gitnet.fr/deblan/side_menu/issues/new?template=.gitea%2fissue_template%2fISSUE_TEMPLATE.yml"
rel="noopener"
>
<NcButton variant="secondary">{{ trans('Report a bug') }}</NcButton>
</a>
<NcButton
variant="secondary"
@click="showConfig = true"
>{{ trans('Show the configuration') }}</NcButton
>
<NcModal
v-if="showConfig"
@close="showConfig = false"
>
<div class="modal__content">
<p style="margin-bottom: 5px">{{ trans('Configuration:') }}</p>
<textarea
class="config"
readonly
v-text="filterConfig(config)"
></textarea>
<div class="modal__footer">
<NcButton
variant="secondary"
@click="copyConfig"
>
<span v-if="configCopied">{{ trans('Done!') }}</span>
<span v-else>{{ trans('Copy') }}</span>
</NcButton>
<NcButton
variant="primary"
@click="showConfig = false"
>
{{ t('side_menu', 'Close') }}
</NcButton>
</div>
</div>
</NcModal>
</TableValue>
</TableRow>
</TableContainer>
</NcAppContent>
</NcContent>
</template>
<script setup>
import { NcContent, NcAppContent, NcButton, NcModal } from '@nextcloud/vue'
import { ref, onMounted } from 'vue'
import { useConfigStore } from '../store/config.js'
import TableContainer from '../components/settings/TableContainer'
import TableRow from '../components/settings/TableRow'
import TableLabel from '../components/settings/TableLabel'
import TableValue from '../components/settings/TableValue'
import SectionTitle from '../components/settings/SectionTitle'
import FormRange from '../components/settings/form/FormRange'
import FormColorPicker from '../components/settings/form/FormColorPicker'
import FormOpener from '../components/settings/form/FormOpener'
import FormSelect from '../components/settings/form/FormSelect'
import FormYesNo from '../components/settings/form/FormYesNo'
import FormSize from '../components/settings/form/FormSize'
import FormAppPicker from '../components/settings/form/FormAppPicker'
import FormAppSort from '../components/settings/form/FormAppSort'
import FormDisplayPicker from '../components/settings/form/FormDisplayPicker'
const menu = [
{ label: 'Global', section: 'global' },
{ label: 'Panel', section: 'panel' },
{ label: 'Top menu', section: 'topMenu' },
{ label: 'Colors', section: 'colors' },
{ label: 'Opener', section: 'opener' },
{ label: 'Applications', section: 'apps' },
{ label: 'Support', section: 'support' },
]
const config = ref(null)
const showConfig = ref(false)
const configCopied = ref(false)
const configStore = useConfigStore()
const section = ref('apps' /*menu[0].section*/)
const setSection = (value) => {
section.value = value
}
const trans = (value) => {
return t('side_menu', value)
}
const sectionClass = (value) => {
return {
hidden: value !== section.value,
}
}
const copyConfig = () => {
navigator.clipboard.writeText(JSON.stringify(filterConfig(config.value), null, 2))
configCopied.value = true
window.setTimeout(() => {
configCopied.value = false
}, 2000)
}
const filterConfig = (value) => {
const result = {}
for (let key in value) {
if (['cache-categories', 'cache'].includes(key) === false) {
result[key] = value[key]
}
}
return result
}
onMounted(async () => {
config.value = await configStore.getAppConfig()
})
</script>
<style scoped>
.hidden {
display: none;
}
.button-inline button {
display: inline-block;
margin-right: 5px;
}
.config {
width: 100%;
height: 30vh;
}
.modal__content {
padding: 20px;
}
.modal__footer {
margin-top: 20px;
text-align: right;
}
.modal__footer button {
display: inline-block;
margin-right: 5px;
}
</style>