forked from platypush/platypush
[Integrations UI] Added new Integrations
panel.
This commit is contained in:
parent
0055acad9d
commit
398d64c53f
11 changed files with 436 additions and 80 deletions
|
@ -4,7 +4,7 @@ import Clipboard from "@/utils/Clipboard";
|
||||||
import Cookies from "@/utils/Cookies";
|
import Cookies from "@/utils/Cookies";
|
||||||
import DateTime from "@/utils/DateTime";
|
import DateTime from "@/utils/DateTime";
|
||||||
import Events from "@/utils/Events";
|
import Events from "@/utils/Events";
|
||||||
import Integrations from "@/utils/Integrations";
|
import Extensions from "@/utils/Extensions";
|
||||||
import Notification from "@/utils/Notification";
|
import Notification from "@/utils/Notification";
|
||||||
import Screen from "@/utils/Screen";
|
import Screen from "@/utils/Screen";
|
||||||
import Text from "@/utils/Text";
|
import Text from "@/utils/Text";
|
||||||
|
@ -19,7 +19,7 @@ export default {
|
||||||
DateTime,
|
DateTime,
|
||||||
Events,
|
Events,
|
||||||
Notification,
|
Notification,
|
||||||
Integrations,
|
Extensions,
|
||||||
Screen,
|
Screen,
|
||||||
Text,
|
Text,
|
||||||
Types,
|
Types,
|
||||||
|
|
|
@ -29,6 +29,9 @@
|
||||||
"execute": {
|
"execute": {
|
||||||
"class": "fa fa-play"
|
"class": "fa fa-play"
|
||||||
},
|
},
|
||||||
|
"extensions": {
|
||||||
|
"class": "fas fa-puzzle-piece"
|
||||||
|
},
|
||||||
"light.hue": {
|
"light.hue": {
|
||||||
"class": "fas fa-lightbulb"
|
"class": "fas fa-lightbulb"
|
||||||
},
|
},
|
||||||
|
|
|
@ -101,10 +101,11 @@ export default {
|
||||||
return names
|
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, 'execute')
|
||||||
panelNames = prepend(panelNames, 'entities')
|
panelNames = prepend(panelNames, 'entities')
|
||||||
return panelNames.sort()
|
return panelNames
|
||||||
},
|
},
|
||||||
|
|
||||||
collapsedDefault() {
|
collapsedDefault() {
|
||||||
|
@ -125,6 +126,8 @@ export default {
|
||||||
return 'Home'
|
return 'Home'
|
||||||
if (name === 'execute')
|
if (name === 'execute')
|
||||||
return 'Execute'
|
return 'Execute'
|
||||||
|
if (name === 'extensions')
|
||||||
|
return 'Extensions'
|
||||||
|
|
||||||
return name
|
return name
|
||||||
},
|
},
|
||||||
|
@ -252,12 +255,12 @@ nav {
|
||||||
}
|
}
|
||||||
|
|
||||||
.plugins {
|
.plugins {
|
||||||
height: calc(100% - #{$toggler-height} - #{$footer-expanded-height} - 1em);
|
height: calc(100% - #{$toggler-height} - #{$footer-expanded-height} - 1.5em);
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
.footer {
|
.footer {
|
||||||
height: $footer-expanded-height;
|
height: calc($footer-expanded-height + 0.4em);
|
||||||
background: $nav-footer-bg;
|
background: $nav-footer-bg;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
|
|
|
@ -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>
|
|
@ -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>
|
|
@ -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>
|
|
@ -5,7 +5,6 @@
|
||||||
v-if="selectedPanel === 'users' && currentUser" />
|
v-if="selectedPanel === 'users' && currentUser" />
|
||||||
<Token :session-token="sessionToken" :current-user="currentUser"
|
<Token :session-token="sessionToken" :current-user="currentUser"
|
||||||
v-else-if="selectedPanel === 'tokens' && currentUser" />
|
v-else-if="selectedPanel === 'tokens' && currentUser" />
|
||||||
<Integrations v-else-if="selectedPanel === 'integrations'" />
|
|
||||||
</main>
|
</main>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
@ -13,12 +12,11 @@
|
||||||
<script>
|
<script>
|
||||||
import Token from "@/components/panels/Settings/Token";
|
import Token from "@/components/panels/Settings/Token";
|
||||||
import Users from "@/components/panels/Settings/Users";
|
import Users from "@/components/panels/Settings/Users";
|
||||||
import Integrations from "@/components/panels/Settings/Integrations";
|
|
||||||
import Utils from "@/Utils";
|
import Utils from "@/Utils";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "Settings",
|
name: "Settings",
|
||||||
components: {Users, Token, Integrations},
|
components: {Users, Token},
|
||||||
mixins: [Utils],
|
mixins: [Utils],
|
||||||
|
|
||||||
props: {
|
props: {
|
||||||
|
|
|
@ -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>
|
|
|
@ -11,12 +11,5 @@
|
||||||
"icon": {
|
"icon": {
|
||||||
"class": "fas fa-key"
|
"class": "fas fa-key"
|
||||||
}
|
}
|
||||||
},
|
|
||||||
|
|
||||||
"integrations": {
|
|
||||||
"name": "Integrations",
|
|
||||||
"icon": {
|
|
||||||
"class": "fas fa-puzzle-piece"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
<script>
|
<script>
|
||||||
export default {
|
export default {
|
||||||
name: "Integrations",
|
name: "Extensions",
|
||||||
methods: {
|
methods: {
|
||||||
pluginDisplayName(name) {
|
pluginDisplayName(name) {
|
||||||
const words = name.split('.')
|
const words = name.split('.')
|
|
@ -93,8 +93,9 @@ export default {
|
||||||
},
|
},
|
||||||
|
|
||||||
initializeDefaultViews() {
|
initializeDefaultViews() {
|
||||||
this.plugins.execute = {}
|
|
||||||
this.plugins.entities = {}
|
this.plugins.entities = {}
|
||||||
|
this.plugins.execute = {}
|
||||||
|
this.plugins.extensions = {}
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue