Use appsidebar for apps
Signed-off-by: John Molakvoæ (skjnldsv) <skjnldsv@protonmail.com>
This commit is contained in:
parent
1ae1d95d8a
commit
434d2c4751
4 changed files with 170 additions and 124 deletions
|
@ -22,81 +22,7 @@
|
|||
|
||||
<template>
|
||||
<div id="app-details-view" style="padding: 20px;">
|
||||
<h2>
|
||||
<div v-if="!app.preview" class="icon-settings-dark" />
|
||||
<svg v-if="app.previewAsIcon && app.preview"
|
||||
width="32"
|
||||
height="32"
|
||||
viewBox="0 0 32 32">
|
||||
<defs><filter :id="filterId"><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="32"
|
||||
height="32"
|
||||
preserveAspectRatio="xMinYMin meet"
|
||||
:filter="filterUrl"
|
||||
:xlink:href="app.preview"
|
||||
class="app-icon" />
|
||||
</svg>
|
||||
{{ app.name }}
|
||||
</h2>
|
||||
<img v-if="app.screenshot" :src="app.screenshot" width="100%">
|
||||
<div v-if="app.level === 300 || app.level === 200 || hasRating" class="app-level">
|
||||
<span v-if="app.level === 300"
|
||||
v-tooltip.auto="t('settings', 'This app is supported via your current Nextcloud subscription.')"
|
||||
class="supported icon-checkmark-color">
|
||||
{{ t('settings', 'Supported') }}</span>
|
||||
<span v-if="app.level === 200"
|
||||
v-tooltip.auto="t('settings', 'Featured apps are developed by and within the community. They offer central functionality and are ready for production use.')"
|
||||
class="official icon-checkmark">
|
||||
{{ t('settings', 'Featured') }}</span>
|
||||
<AppScore v-if="hasRating" :score="app.appstoreData.ratingOverall" />
|
||||
</div>
|
||||
|
||||
<div v-if="author" class="app-author">
|
||||
{{ t('settings', 'by') }}
|
||||
<span v-for="(a, index) in author" :key="index">
|
||||
<a v-if="a['@attributes'] && a['@attributes']['homepage']" :href="a['@attributes']['homepage']">{{ a['@value'] }}</a><span v-else-if="a['@value']">{{ a['@value'] }}</span><span v-else>{{ a }}</span><span v-if="index+1 < author.length">, </span>
|
||||
</span>
|
||||
</div>
|
||||
<div v-if="licence" class="app-licence">
|
||||
{{ licence }}
|
||||
</div>
|
||||
<div class="actions">
|
||||
<div class="actions-buttons">
|
||||
<input v-if="app.update"
|
||||
class="update primary"
|
||||
type="button"
|
||||
:value="t('settings', 'Update to {version}', {version: app.update})"
|
||||
:disabled="installing || loading(app.id)"
|
||||
@click="update(app.id)">
|
||||
<input v-if="app.canUnInstall"
|
||||
class="uninstall"
|
||||
type="button"
|
||||
:value="t('settings', 'Remove')"
|
||||
:disabled="installing || loading(app.id)"
|
||||
@click="remove(app.id)">
|
||||
<input v-if="app.active"
|
||||
class="enable"
|
||||
type="button"
|
||||
:value="t('settings','Disable')"
|
||||
:disabled="installing || loading(app.id)"
|
||||
@click="disable(app.id)">
|
||||
<input v-if="!app.active && (app.canInstall || app.isCompatible)"
|
||||
v-tooltip.auto="enableButtonTooltip"
|
||||
class="enable primary"
|
||||
type="button"
|
||||
:value="enableButtonText"
|
||||
:disabled="!app.canInstall || installing || loading(app.id)"
|
||||
@click="enable(app.id)">
|
||||
<input v-else-if="!app.active"
|
||||
v-tooltip.auto="forceEnableButtonTooltip"
|
||||
class="enable force"
|
||||
type="button"
|
||||
:value="forceEnableButtonText"
|
||||
:disabled="installing || loading(app.id)"
|
||||
@click="forceEnable(app.id)">
|
||||
</div>
|
||||
<div class="app-groups">
|
||||
<div v-if="app.active && canLimitToGroups(app)" class="groups-enable">
|
||||
<input :id="prefix('groups_enable', app.id)"
|
||||
|
@ -192,9 +118,8 @@ import marked from 'marked'
|
|||
import dompurify from 'dompurify'
|
||||
|
||||
import AppScore from './AppList/AppScore'
|
||||
import AppManagement from './AppManagement'
|
||||
import AppManagement from '../mixins/AppManagement'
|
||||
import PrefixMixin from './PrefixMixin'
|
||||
import SvgFilterMixin from './SvgFilterMixin'
|
||||
|
||||
export default {
|
||||
name: 'AppDetails',
|
||||
|
@ -202,13 +127,8 @@ export default {
|
|||
Multiselect,
|
||||
AppScore,
|
||||
},
|
||||
mixins: [AppManagement, PrefixMixin, SvgFilterMixin],
|
||||
mixins: [AppManagement, PrefixMixin],
|
||||
props: ['category', 'app'],
|
||||
data() {
|
||||
return {
|
||||
groupCheckedAppsData: false,
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
appstoreUrl() {
|
||||
return `https://apps.nextcloud.com/apps/${this.app.id}`
|
||||
|
|
|
@ -69,38 +69,38 @@
|
|||
<div v-if="app.error" class="warning">
|
||||
{{ app.error }}
|
||||
</div>
|
||||
<div v-if="loading(app.id)" class="icon icon-loading-small" />
|
||||
<div v-if="isLoading" class="icon icon-loading-small" />
|
||||
<input v-if="app.update"
|
||||
class="update primary"
|
||||
type="button"
|
||||
:value="t('settings', 'Update to {update}', {update:app.update})"
|
||||
:disabled="installing || loading(app.id)"
|
||||
:disabled="installing || isLoading"
|
||||
@click.stop="update(app.id)">
|
||||
<input v-if="app.canUnInstall"
|
||||
class="uninstall"
|
||||
type="button"
|
||||
:value="t('settings', 'Remove')"
|
||||
:disabled="installing || loading(app.id)"
|
||||
:disabled="installing || isLoading"
|
||||
@click.stop="remove(app.id)">
|
||||
<input v-if="app.active"
|
||||
class="enable"
|
||||
type="button"
|
||||
:value="t('settings','Disable')"
|
||||
:disabled="installing || loading(app.id)"
|
||||
:disabled="installing || isLoading"
|
||||
@click.stop="disable(app.id)">
|
||||
<input v-if="!app.active && (app.canInstall || app.isCompatible)"
|
||||
v-tooltip.auto="enableButtonTooltip"
|
||||
class="enable"
|
||||
type="button"
|
||||
:value="enableButtonText"
|
||||
:disabled="!app.canInstall || installing || loading(app.id)"
|
||||
:disabled="!app.canInstall || installing || isLoading"
|
||||
@click.stop="enable(app.id)">
|
||||
<input v-else-if="!app.active"
|
||||
v-tooltip.auto="forceEnableButtonTooltip"
|
||||
class="enable force"
|
||||
type="button"
|
||||
:value="forceEnableButtonText"
|
||||
:disabled="installing || loading(app.id)"
|
||||
:disabled="installing || isLoading"
|
||||
@click.stop="forceEnable(app.id)">
|
||||
</div>
|
||||
</div>
|
||||
|
@ -108,7 +108,7 @@
|
|||
|
||||
<script>
|
||||
import AppScore from './AppScore'
|
||||
import AppManagement from '../AppManagement'
|
||||
import AppManagement from '../../mixins/AppManagement'
|
||||
import SvgFilterMixin from '../SvgFilterMixin'
|
||||
|
||||
export default {
|
||||
|
|
|
@ -1,40 +1,37 @@
|
|||
<!--
|
||||
- @copyright Copyright (c) 2018 Julius Härtl <jus@bitgrid.net>
|
||||
-
|
||||
- @author Julius Härtl <jus@bitgrid.net>
|
||||
-
|
||||
- @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/>.
|
||||
-
|
||||
-->
|
||||
/**
|
||||
* @copyright Copyright (c) 2019 Julius Härtl <jus@bitgrid.net>
|
||||
*
|
||||
* @author Julius Härtl <jus@bitgrid.net>
|
||||
* @author John Molakvoæ <skjnldsv@protonmail.com>
|
||||
*
|
||||
* @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/>.
|
||||
*
|
||||
*/
|
||||
|
||||
<script>
|
||||
export default {
|
||||
computed: {
|
||||
appGroups() {
|
||||
return this.app.groups.map(group => { return { id: group, name: group } })
|
||||
},
|
||||
loading() {
|
||||
const self = this
|
||||
return function(id) {
|
||||
return self.$store.getters.loading(id)
|
||||
}
|
||||
},
|
||||
installing() {
|
||||
return this.$store.getters.loading('install')
|
||||
},
|
||||
isLoading() {
|
||||
return this.app && this.$store.getters.loading(this.app.id)
|
||||
},
|
||||
enableButtonText() {
|
||||
if (this.app.needsDownload) {
|
||||
return t('settings', 'Download and enable')
|
||||
|
@ -61,11 +58,19 @@ export default {
|
|||
return base
|
||||
},
|
||||
},
|
||||
|
||||
data() {
|
||||
return {
|
||||
groupCheckedAppsData: false,
|
||||
}
|
||||
},
|
||||
|
||||
mounted() {
|
||||
if (this.app.groups.length > 0) {
|
||||
if (this.app && this.app.groups && this.app.groups.length > 0) {
|
||||
this.groupCheckedAppsData = true
|
||||
}
|
||||
},
|
||||
|
||||
methods: {
|
||||
asyncFindGroup(query) {
|
||||
return this.$store.dispatch('getGroups', { search: query, limit: 5, offset: 0 })
|
||||
|
@ -135,4 +140,3 @@ export default {
|
|||
},
|
||||
},
|
||||
}
|
||||
</script>
|
|
@ -22,7 +22,7 @@
|
|||
|
||||
<template>
|
||||
<Content app-name="settings"
|
||||
:class="{ 'with-app-sidebar': currentApp}"
|
||||
:class="{ 'with-app-sidebar': app}"
|
||||
:content-class="{ 'icon-loading': loadingList }"
|
||||
:navigation-class="{ 'icon-loading': loading }">
|
||||
<AppNavigation>
|
||||
|
@ -81,15 +81,70 @@
|
|||
</ul>
|
||||
</AppNavigation>
|
||||
<AppContent class="app-settings-content" :class="{ 'icon-loading': loadingList }">
|
||||
<AppList :category="category" :app="currentApp" :search="searchQuery" />
|
||||
<AppList :category="category" :app="app" :search="searchQuery" />
|
||||
</AppContent>
|
||||
<AppSidebar v-if="id && currentApp" @close="hideAppDetails">
|
||||
<AppDetails :category="category" :app="currentApp" />
|
||||
<AppSidebar
|
||||
v-if="id && app"
|
||||
v-bind="appSidebar"
|
||||
:class="{'app-sidebar--without-background': !appSidebar.background}"
|
||||
@close="hideAppDetails">
|
||||
<template v-if="!appSidebar.background" #header>
|
||||
<div class="app-sidebar-header__figure--default-app-icon icon-settings-dark" />
|
||||
</template>
|
||||
<template #primary-actions>
|
||||
<div v-if="app.level === 300 || app.level === 200 || hasRating" class="app-level">
|
||||
<span v-if="app.level === 300"
|
||||
v-tooltip.auto="t('settings', 'This app is supported via your current Nextcloud subscription.')"
|
||||
class="supported icon-checkmark-color">
|
||||
{{ t('settings', 'Supported') }}</span>
|
||||
<span v-if="app.level === 200"
|
||||
v-tooltip.auto="t('settings', 'Featured apps are developed by and within the community. They offer central functionality and are ready for production use.')"
|
||||
class="official icon-checkmark">
|
||||
{{ t('settings', 'Featured') }}</span>
|
||||
<AppScore v-if="hasRating" :score="app.appstoreData.ratingOverall" />
|
||||
</div>
|
||||
</template>
|
||||
<template #secondary-actions>
|
||||
<ActionButton v-if="app.update"
|
||||
:disabled="installing || isLoading"
|
||||
icon="icon-download"
|
||||
@click="update(app.id)">
|
||||
{{ t('settings', 'Update to {version}', {version: app.update}) }}
|
||||
</ActionButton>
|
||||
<ActionButton v-if="app.canUnInstall"
|
||||
:disabled="installing || isLoading"
|
||||
icon="icon-delete"
|
||||
@click="remove(app.id)">
|
||||
{{ t('settings', 'Remove') }}
|
||||
</ActionButton>
|
||||
<ActionButton v-if="app.active"
|
||||
:disabled="installing || isLoading"
|
||||
icon="icon-close"
|
||||
@click="disable(app.id)">
|
||||
{{ t('settings','Disable') }}
|
||||
</ActionButton>
|
||||
<ActionButton v-if="!app.active && (app.canInstall || app.isCompatible)"
|
||||
v-tooltip.auto="enableButtonTooltip"
|
||||
:disabled="!app.canInstall || installing || isLoading"
|
||||
icon="icon-checkmark"
|
||||
@click="enable(app.id)">
|
||||
{{ enableButtonText }}
|
||||
</ActionButton>
|
||||
<ActionButton v-else-if="!app.active"
|
||||
v-tooltip.auto="forceEnableButtonTooltip"
|
||||
:disabled="installing || isLoading"
|
||||
icon="icon-checkmark"
|
||||
@click="forceEnable(app.id)">
|
||||
{{ forceEnableButtonText }}
|
||||
</ActionButton>
|
||||
</template>
|
||||
<AppDetails :category="category" :app="app" />
|
||||
</AppSidebar>
|
||||
</Content>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import ActionButton from '@nextcloud/vue/dist/Components/ActionButton'
|
||||
import AppContent from '@nextcloud/vue/dist/Components/AppContent'
|
||||
import AppNavigation from '@nextcloud/vue/dist/Components/AppNavigation'
|
||||
import AppNavigationCounter from '@nextcloud/vue/dist/Components/AppNavigationCounter'
|
||||
|
@ -102,12 +157,14 @@ import VueLocalStorage from 'vue-localstorage'
|
|||
|
||||
import AppList from '../components/AppList'
|
||||
import AppDetails from '../components/AppDetails'
|
||||
import AppManagement from '../mixins/AppManagement'
|
||||
|
||||
Vue.use(VueLocalStorage)
|
||||
|
||||
export default {
|
||||
name: 'Apps',
|
||||
components: {
|
||||
ActionButton,
|
||||
AppContent,
|
||||
AppDetails,
|
||||
AppList,
|
||||
|
@ -118,6 +175,7 @@ export default {
|
|||
AppSidebar,
|
||||
Content,
|
||||
},
|
||||
mixins: [AppManagement],
|
||||
props: {
|
||||
category: {
|
||||
type: String,
|
||||
|
@ -140,7 +198,7 @@ export default {
|
|||
loadingList() {
|
||||
return this.$store.getters.loading('list')
|
||||
},
|
||||
currentApp() {
|
||||
app() {
|
||||
return this.apps.find(app => app.id === this.id)
|
||||
},
|
||||
categories() {
|
||||
|
@ -155,6 +213,30 @@ export default {
|
|||
settings() {
|
||||
return this.$store.getters.getServerData
|
||||
},
|
||||
|
||||
// sidebar app binding
|
||||
appSidebar() {
|
||||
const author = Array.isArray(this.app.author)
|
||||
? this.app.author[0]['@value']
|
||||
? this.app.author.map(author => author['@value']).join(', ')
|
||||
: this.app.author.join(', ')
|
||||
: this.app.author['@value']
|
||||
? this.app.author['@value']
|
||||
: this.app.author
|
||||
const license = t('settings', '{license}-licensed', { license: ('' + this.app.licence).toUpperCase() })
|
||||
|
||||
const subtitle = t('settings', 'by {author}\n{license}', { author, license })
|
||||
|
||||
return {
|
||||
subtitle,
|
||||
background: this.app.screenshot
|
||||
? this.app.screenshot
|
||||
: this.app.preview,
|
||||
compact: !this.app.screenshot,
|
||||
title: this.app.name,
|
||||
|
||||
}
|
||||
},
|
||||
},
|
||||
watch: {
|
||||
category: function(val, old) {
|
||||
|
@ -189,3 +271,43 @@ export default {
|
|||
},
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
|
||||
#app-sidebar::v-deep {
|
||||
&:not(.app-sidebar--without-background) {
|
||||
// with full screenshot, let's fill the figure
|
||||
:not(.app-sidebar-header--compact) .app-sidebar-header__figure {
|
||||
background-size: cover
|
||||
}
|
||||
// revert sidebar app icon so it is black
|
||||
.app-sidebar-header--compact .app-sidebar-header__figure {
|
||||
filter: invert(1);
|
||||
background-size: 32px;
|
||||
}
|
||||
}
|
||||
|
||||
// default icon slot styling
|
||||
&.app-sidebar--without-background {
|
||||
.app-sidebar-header__figure {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
&--default-app-icon {
|
||||
height: 32px;
|
||||
width: 32px;
|
||||
background-size: 32px;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// allow multi line subtitle for the license
|
||||
.app-sidebar-header__subtitle {
|
||||
white-space: pre-line !important;
|
||||
line-height: 16px;
|
||||
overflow: visible !important;
|
||||
height: 22px;
|
||||
}
|
||||
}
|
||||
|
||||
</style>
|
||||
|
|
Loading…
Reference in a new issue