[Integrations UI] Added new Integrations panel.

This commit is contained in:
Fabio Manganiello 2023-10-16 01:14:19 +02:00
parent 0055acad9d
commit 398d64c53f
Signed by untrusted user: blacklight
GPG key ID: D90FBA7F76362774
11 changed files with 436 additions and 80 deletions

View file

@ -4,7 +4,7 @@ import Clipboard from "@/utils/Clipboard";
import Cookies from "@/utils/Cookies";
import DateTime from "@/utils/DateTime";
import Events from "@/utils/Events";
import Integrations from "@/utils/Integrations";
import Extensions from "@/utils/Extensions";
import Notification from "@/utils/Notification";
import Screen from "@/utils/Screen";
import Text from "@/utils/Text";
@ -19,7 +19,7 @@ export default {
DateTime,
Events,
Notification,
Integrations,
Extensions,
Screen,
Text,
Types,

View file

@ -29,6 +29,9 @@
"execute": {
"class": "fa fa-play"
},
"extensions": {
"class": "fas fa-puzzle-piece"
},
"light.hue": {
"class": "fas fa-lightbulb"
},

View file

@ -101,10 +101,11 @@ export default {
return names
}
let panelNames = Object.keys(this.panels)
let panelNames = Object.keys(this.panels).sort()
panelNames = prepend(panelNames, 'extensions')
panelNames = prepend(panelNames, 'execute')
panelNames = prepend(panelNames, 'entities')
return panelNames.sort()
return panelNames
},
collapsedDefault() {
@ -125,6 +126,8 @@ export default {
return 'Home'
if (name === 'execute')
return 'Execute'
if (name === 'extensions')
return 'Extensions'
return name
},
@ -252,12 +255,12 @@ nav {
}
.plugins {
height: calc(100% - #{$toggler-height} - #{$footer-expanded-height} - 1em);
height: calc(100% - #{$toggler-height} - #{$footer-expanded-height} - 1.5em);
overflow: auto;
}
.footer {
height: $footer-expanded-height;
height: calc($footer-expanded-height + 0.4em);
background: $nav-footer-bg;
padding: 0;
margin: 0;

View file

@ -0,0 +1,118 @@
<template>
<section class="doc">
<header>
<h2>
<a class="title" :href="extension.doc_url" target="_blank">
<i class="icon fas fa-book" />
{{ extension.name }}
</a>
</h2>
</header>
<article v-html="doc" v-if="doc" @click="onDocClick" />
</section>
</template>
<script>
import Utils from "@/Utils"
import { bus } from "@/bus";
export default {
name: "Doc",
mixins: [Utils],
props: {
extension: {
type: Object,
required: true,
},
},
data() {
return {
doc: null,
}
},
methods: {
async parseDoc() {
if (!this.extension.doc?.length)
return null
return await this.request(
'utils.rst_to_html',
{text: this.extension.doc}
)
},
refreshDoc() {
this.parseDoc().then(doc => this.doc = doc)
},
// Intercept links to the documentation and replace them with
// in-app connections, or opens them in a new tab if they
// don't point to an internal documentation page.
onDocClick(event) {
if (!event.target.tagName.toLowerCase() === 'a')
return
event.preventDefault()
const href = event.target.getAttribute('href')
if (!href)
return
const match = href.match(/^https:\/\/docs\.platypush\.tech\/platypush\/(plugins|backend)\/([\w.]+)\.html#?.*$/)
if (!match) {
event.preventDefault()
window.open(href, '_blank')
return
}
let [_, type, name] = match
if (type === 'backend')
name = `backend.${name}`
bus.emit('update:extension', name)
event.preventDefault()
},
},
mounted() {
this.refreshDoc()
this.$watch('extension.doc', this.refreshDoc)
},
}
</script>
<style lang="scss" scoped>
$header-height: 3em;
section {
height: 100%;
header {
height: $header-height;
padding: 0.5em;
border-bottom: 1px solid $border-color-2;
h2 {
margin: 0;
padding: 0;
font-size: 1.25em;
}
}
article {
height: calc(100% - #{$header-height});
padding: 0.5em;
overflow: auto;
:deep(ul) {
margin-left: 1em;
li {
list-style: disc;
}
}
}
}
</style>

View file

@ -0,0 +1,91 @@
<template>
<div class="extension">
<header>
<Tabs>
<Tab :selected="selectedTab === 'doc'" icon-class="fas fa-book"
@input="selectedTab = 'doc'">
<span class="from tablet">Documentation</span>
</Tab>
<Tab :selected="selectedTab === 'install'" icon-class="fas fa-download"
@input="selectedTab = 'install'">
<span class="from tablet">Install</span>
</Tab>
<Tab :selected="selectedTab === 'conf'" icon-class="fas fa-square-check"
@input="selectedTab = 'conf'">
<span class="from tablet">Configuration</span>
</Tab>
<Tab :selected="selectedTab === 'actions'" icon-class="fas fa-play"
@input="selectedTab = 'actions'">
<span class="from tablet">Actions</span>
</Tab>
</Tabs>
</header>
<div class="extension-body">
<Doc v-if="selectedTab === 'doc'" :extension="extension" />
</div>
</div>
</template>
<script>
import Tab from "@/components/elements/Tab"
import Tabs from "@/components/elements/Tabs"
import Doc from "./Doc"
export default {
name: "Extension",
components: {
Doc,
Tab,
Tabs,
},
props: {
extension: {
type: Object,
required: true,
},
},
data() {
return {
selectedTab: 'doc',
}
},
}
</script>
<style lang="scss" scoped>
@import "src/style/items";
$header-height: 4em;
.extension {
width: 100%;
height: 100%;
background: $background-color;
display: flex;
flex-direction: column;
border-top: 1px solid $border-color-1;
box-shadow: $border-shadow-bottom;
header {
height: $header-height;
:deep(.tabs) {
margin: 0;
}
}
.extension-body {
height: calc(100% - #{$header-height});
overflow: auto;
:deep(section) {
height: calc(100% - #{$header-height});
}
}
}
</style>

View file

@ -0,0 +1,211 @@
<template>
<div class="row plugin extensions-container">
<Loading v-if="loading" />
<header>
<div class="filter-container">
<input type="text"
ref="filter"
placeholder="Extension name"
v-model="filter"
:disabled="loading" />
</div>
</header>
<main>
<div class="items">
<div class="extension-container" v-for="name in extensionNames" :key="name">
<div class="extension" v-if="matchesFilter(name)">
<div class="item name"
:class="{selected: name === selectedExtension}"
:data-name="name"
@click="onInput(name, false)"
v-text="extensions[name].name" />
<div class="extension-body-container until tablet"
v-if="selectedExtension && name === selectedExtension">
<Extension :extension="extensions[selectedExtension]" />
</div>
</div>
</div>
</div>
<div class="extension-body-container from desktop"
v-if="selectedExtension">
<Extension :extension="extensions[selectedExtension]" />
</div>
</main>
</div>
</template>
<script>
import Loading from "@/components/Loading"
import Utils from "@/Utils"
import Extension from "./Extension"
import { bus } from "@/bus";
export default {
name: "Extensions",
mixins: [Utils],
components: {
Extension,
Loading,
Utils,
},
data() {
return {
loading: false,
plugins: {},
backends: {},
filter: '',
selectedExtension: null,
}
},
computed: {
extensions() {
const extensions = {}
Object.entries(this.plugins).forEach(([name, plugin]) => {
extensions[name] = {
...plugin,
name: name,
}
})
Object.entries(this.backends).forEach(([name, backend]) => {
name = `backend.${name}`
extensions[name] = {
...backend,
name: name,
}
})
return extensions
},
extensionNames() {
return Object.keys(this.extensions).sort()
},
},
methods: {
onInput(input, setFilter = true) {
if (setFilter) {
this.filter = input
}
const name = input?.toLowerCase()?.trim()
if (name?.length && name !== this.selectedExtension && this.extensions[name]) {
this.selectedExtension = name
const el = this.$el.querySelector(`.extensions-container .item[data-name="${name}"]`)
if (el)
el.scrollIntoView({behavior: 'smooth'})
} else {
this.selectedExtension = null
}
},
matchesFilter(extension) {
if (!this.filter) {
return true
}
return extension.includes(this.filter.toLowerCase())
},
async loadExtensions() {
this.loading = true
try {
[this.plugins, this.backends] =
await Promise.all([
this.request('inspect.get_all_plugins'),
this.request('inspect.get_all_backends'),
])
} finally {
this.loading = false
}
},
},
mounted() {
this.loadExtensions()
bus.on('update:extension', (ext) => this.onInput(ext, false))
this.$nextTick(() => this.$refs.filter.focus())
}
}
</script>
<style lang="scss" scoped>
@import "src/style/items";
@import "../Execute/common";
$header-height: 3.25em;
.extensions-container {
width: 100%;
display: flex;
flex-direction: column;
margin-top: .15em;
header {
height: $header-height;
padding: 0.5em;
margin-bottom: 2px;
box-shadow: $border-shadow-bottom;
.filter-container {
width: 100%;
input {
width: 100%;
}
}
}
main {
height: calc(100% - #{$header-height} - 0.25em);
min-height: calc(100% - #{$header-height} - 0.25em);
background: $background-color;
display: flex;
flex-direction: row;
}
.items {
height: 100%;
flex-grow: 1;
overflow: auto;
border-bottom: $default-border-2;
}
.extension-container {
.extension {
display: flex;
flex-direction: column;
.name {
padding: 1em;
&.selected {
font-weight: bold;
}
}
}
}
.extension-body-container.desktop {
width: 70%;
height: 100%;
min-height: 100%;
border-left: $default-border-2;
border-bottom: $default-border-2;
:deep(article) {
height: 100%;
overflow: auto;
}
}
}
</style>

View file

@ -5,7 +5,6 @@
v-if="selectedPanel === 'users' && currentUser" />
<Token :session-token="sessionToken" :current-user="currentUser"
v-else-if="selectedPanel === 'tokens' && currentUser" />
<Integrations v-else-if="selectedPanel === 'integrations'" />
</main>
</div>
</template>
@ -13,12 +12,11 @@
<script>
import Token from "@/components/panels/Settings/Token";
import Users from "@/components/panels/Settings/Users";
import Integrations from "@/components/panels/Settings/Integrations";
import Utils from "@/Utils";
export default {
name: "Settings",
components: {Users, Token, Integrations},
components: {Users, Token},
mixins: [Utils],
props: {

View file

@ -1,62 +0,0 @@
<template>
<div class="integrations-container">
<Loading v-if="loading" />
<div class="body">
<!-- TODO -->
</div>
</div>
</template>
<script>
import Loading from "@/components/Loading";
import Utils from "@/Utils";
export default {
name: "Integrations",
components: {Loading},
mixins: [Utils],
data() {
return {
loading: false,
plugins: {},
backends: {},
}
},
methods: {
async loadIntegrations() {
this.loading = true
try {
[this.plugins, this.backends] =
await Promise.all([
this.request('inspect.get_all_plugins'),
this.request('inspect.get_all_backends'),
])
} finally {
this.loading = false
}
},
},
mounted() {
this.loadIntegrations()
}
}
</script>
<style lang="scss">
.integrations-container {
width: 100%;
display: flex;
flex-direction: column;
margin-top: .15em;
.body {
background: $background-color;
display: flex;
}
}
</style>

View file

@ -11,12 +11,5 @@
"icon": {
"class": "fas fa-key"
}
},
"integrations": {
"name": "Integrations",
"icon": {
"class": "fas fa-puzzle-piece"
}
}
}

View file

@ -1,6 +1,6 @@
<script>
export default {
name: "Integrations",
name: "Extensions",
methods: {
pluginDisplayName(name) {
const words = name.split('.')

View file

@ -93,8 +93,9 @@ export default {
},
initializeDefaultViews() {
this.plugins.execute = {}
this.plugins.entities = {}
this.plugins.execute = {}
this.plugins.extensions = {}
},
},