ShortCuts und Versionen hinzugefügt.
Code leicht aufgeräumt und verbessert.
This commit is contained in:
parent
a23a17285b
commit
4324681b75
|
@ -2,22 +2,29 @@ import { boot } from 'quasar/wrappers';
|
|||
import { RouteConfig } from 'vue-router';
|
||||
import { Module } from 'vuex';
|
||||
|
||||
interface PluginRouteConfig extends RouteConfig {
|
||||
example?: unknown;
|
||||
}
|
||||
const config = {
|
||||
// Do not change required Modules !!
|
||||
requiredModules: ['user'],
|
||||
// here you can import plugins.
|
||||
loadModules: ['plugin1', 'user-plugin']
|
||||
};
|
||||
|
||||
interface EssentialLink {
|
||||
title: string;
|
||||
// do not change anything here !!
|
||||
|
||||
interface ShortCutLink {
|
||||
link: string;
|
||||
icon?: string;
|
||||
icon: string;
|
||||
}
|
||||
|
||||
interface Plugin {
|
||||
name: string;
|
||||
routes: PluginRouteConfig[] | RouteConfig[];
|
||||
routes: RouteConfig[];
|
||||
store?: Module<never, never>[];
|
||||
mainLink: PluginMainLink;
|
||||
requiredModules: string[];
|
||||
shortCuts: ShortCutLink[];
|
||||
shortCutsOut: ShortCutLink[];
|
||||
version: string;
|
||||
}
|
||||
|
||||
interface PluginMainLink extends PluginChildLink {
|
||||
|
@ -31,15 +38,29 @@ interface PluginChildLink {
|
|||
icon: string;
|
||||
}
|
||||
|
||||
const config = {
|
||||
// Do not change required Modules !!
|
||||
requiredModules: ['user'],
|
||||
// here you can import plugins.
|
||||
loadModules: ['plugin1', 'user-plugin']
|
||||
interface LoadedPlugin {
|
||||
name: string;
|
||||
version: string;
|
||||
}
|
||||
|
||||
interface LoadedPlugins {
|
||||
plugins: LoadedPlugin[];
|
||||
routes: RouteConfig[];
|
||||
mainLinks: PluginMainLink[];
|
||||
shortcuts: ShortCutLink[];
|
||||
shortcutsOut: ShortCutLink[];
|
||||
}
|
||||
|
||||
export {
|
||||
Plugin,
|
||||
PluginChildLink,
|
||||
PluginMainLink,
|
||||
ShortCutLink,
|
||||
LoadedPlugins,
|
||||
LoadedPlugin
|
||||
};
|
||||
|
||||
export { PluginRouteConfig, Plugin, PluginChildLink, PluginMainLink };
|
||||
|
||||
// combine routes from source to target
|
||||
function combineRoutes(
|
||||
target: RouteConfig[],
|
||||
source: RouteConfig[]
|
||||
|
@ -52,6 +73,7 @@ function combineRoutes(
|
|||
}
|
||||
);
|
||||
if (targetRouteConfig) {
|
||||
// if exists first layer in target exist iterate through 2nd layer e.g. /main/user, /main/about
|
||||
sourceRouteConfig.children?.forEach(
|
||||
(sourcePluginChildRouteConfig: RouteConfig) => {
|
||||
const targetPluginRouteConfig:
|
||||
|
@ -62,16 +84,20 @@ function combineRoutes(
|
|||
}
|
||||
);
|
||||
if (targetPluginRouteConfig) {
|
||||
// if 2nd layer in target exist check if target has children path.
|
||||
if (targetPluginRouteConfig.children) {
|
||||
// if target has children path, add children from source.
|
||||
targetPluginRouteConfig.children = Object.assign(
|
||||
targetPluginRouteConfig.children,
|
||||
sourcePluginChildRouteConfig.children
|
||||
);
|
||||
} else {
|
||||
// if not set children of targen from children of source
|
||||
targetPluginRouteConfig.children =
|
||||
sourcePluginChildRouteConfig.children;
|
||||
}
|
||||
} else {
|
||||
// if target not exists in 2nd layer, add source to children of first targen
|
||||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
||||
// @ts-ignore
|
||||
targetRouteConfig.children.push(sourcePluginChildRouteConfig);
|
||||
|
@ -79,90 +105,105 @@ function combineRoutes(
|
|||
}
|
||||
);
|
||||
} else {
|
||||
// if target not exists in first layer, add source
|
||||
target.push(sourceRouteConfig);
|
||||
}
|
||||
});
|
||||
return target;
|
||||
}
|
||||
|
||||
// combine Links of Plugins from source to target
|
||||
function combineMainLinks(
|
||||
target: PluginMainLink[],
|
||||
source: PluginMainLink
|
||||
): PluginMainLink[] {
|
||||
console.log('target', target);
|
||||
console.log('source', source);
|
||||
const targetPluginMainLink: PluginMainLink | undefined = target.find(
|
||||
(targetPluginMainLink: PluginMainLink) => {
|
||||
console.log(targetPluginMainLink.title, source.title);
|
||||
return targetPluginMainLink.title == source.title;
|
||||
}
|
||||
);
|
||||
console.log('targetPluginMainLink', targetPluginMainLink);
|
||||
if (targetPluginMainLink) {
|
||||
source.children.forEach((sourcePluginChildLink: PluginChildLink) => {
|
||||
targetPluginMainLink.children.push(sourcePluginChildLink);
|
||||
});
|
||||
} else {
|
||||
console.log('push source', source);
|
||||
target.push(source);
|
||||
}
|
||||
console.log('merged', target);
|
||||
return target;
|
||||
}
|
||||
|
||||
// loade plugins
|
||||
function loadPlugin(
|
||||
loadedPlugins: LoadedPlugins,
|
||||
modules: string[],
|
||||
plugins: Plugin[],
|
||||
store: any
|
||||
): LoadedPlugins {
|
||||
modules.forEach(requiredModule => {
|
||||
const plugin = plugins.find(plugin => {
|
||||
return plugin.name == requiredModule;
|
||||
});
|
||||
if (plugin) {
|
||||
loadedPlugins.routes = combineRoutes(loadedPlugins.routes, plugin.routes);
|
||||
if (plugin.store) {
|
||||
plugin.store.forEach(store_module => {
|
||||
// eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access
|
||||
store.registerModule(store_module);
|
||||
});
|
||||
}
|
||||
loadedPlugins.mainLinks = combineMainLinks(
|
||||
loadedPlugins.mainLinks,
|
||||
plugin.mainLink
|
||||
);
|
||||
loadedPlugins.shortcuts.concat(plugin.shortCuts);
|
||||
loadedPlugins.shortcutsOut.concat(plugin.shortCutsOut);
|
||||
loadedPlugins.plugins.push({
|
||||
name: plugin.name,
|
||||
version: plugin.version
|
||||
});
|
||||
} else {
|
||||
console.exception(`Don't find required Plugin ${requiredModule}`);
|
||||
}
|
||||
});
|
||||
return loadedPlugins;
|
||||
}
|
||||
|
||||
// "async" is optional;
|
||||
// more info on params: https://quasar.dev/quasar-cli/cli-documentation/boot-files#Anatomy-of-a-boot-file
|
||||
export default boot(({ Vue, router, store }) => {
|
||||
const pluginsContext = require.context('src/plugins', true, /.+\/plugin.ts$/);
|
||||
let pluginMainLinks: PluginMainLink[] = [];
|
||||
const plugins: Plugin[] = [];
|
||||
let loadedPlugins: LoadedPlugins = {
|
||||
routes: [],
|
||||
plugins: [],
|
||||
mainLinks: [],
|
||||
shortcuts: [],
|
||||
shortcutsOut: []
|
||||
};
|
||||
|
||||
// get all plugins
|
||||
const pluginsContext = require.context('src/plugins', true, /.+\/plugin.ts$/);
|
||||
pluginsContext.keys().forEach((fileName: string) => {
|
||||
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
|
||||
plugins.push(pluginsContext(fileName).default);
|
||||
});
|
||||
|
||||
let routes: RouteConfig[] = [];
|
||||
// load plugins
|
||||
loadedPlugins = loadPlugin(
|
||||
loadedPlugins,
|
||||
config.requiredModules,
|
||||
plugins,
|
||||
store
|
||||
);
|
||||
loadedPlugins = loadPlugin(loadedPlugins, config.loadModules, plugins, store);
|
||||
|
||||
config.requiredModules.forEach(requiredModule => {
|
||||
const plugin = plugins.find(plugin => {
|
||||
return plugin.name == requiredModule;
|
||||
});
|
||||
if (plugin) {
|
||||
routes = combineRoutes(routes, plugin.routes);
|
||||
if (plugin.store) {
|
||||
plugin.store.forEach(store_module => {
|
||||
// eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access
|
||||
store.registerModule(store_module);
|
||||
});
|
||||
}
|
||||
pluginMainLinks = combineMainLinks(pluginMainLinks, plugin.mainLink);
|
||||
} else {
|
||||
console.exception(`Don't find required Plugin ${requiredModule}`);
|
||||
}
|
||||
});
|
||||
console.log(loadedPlugins.routes);
|
||||
|
||||
config.loadModules.forEach(loadModule => {
|
||||
const plugin = plugins.find(plugin => {
|
||||
return plugin.name == loadModule;
|
||||
});
|
||||
if (plugin) {
|
||||
routes = combineRoutes(routes, plugin.routes);
|
||||
if (plugin.store) {
|
||||
plugin.store.forEach(store_module => {
|
||||
// eslint-disable-next-line @typescript-eslint/no-unsafe-call,@typescript-eslint/no-unsafe-member-access
|
||||
store.registerModule(store_module);
|
||||
});
|
||||
}
|
||||
pluginMainLinks = combineMainLinks(pluginMainLinks, plugin.mainLink);
|
||||
} else {
|
||||
console.exception(`Don't find Plugin ${loadModule}`);
|
||||
}
|
||||
});
|
||||
console.log(routes);
|
||||
router.addRoutes(routes);
|
||||
// add new routes for plugins
|
||||
router.addRoutes(loadedPlugins.routes);
|
||||
|
||||
// save plugins in VM-variable
|
||||
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
|
||||
Vue.prototype.$flaschengeistPluginsMainLinks = pluginMainLinks;
|
||||
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
|
||||
Vue.prototype.$flaschengeistPlugins = plugins;
|
||||
console.log(pluginMainLinks);
|
||||
Vue.prototype.$flaschengeistPlugins = loadedPlugins;
|
||||
console.log(loadedPlugins);
|
||||
});
|
||||
|
|
|
@ -1,59 +0,0 @@
|
|||
<template>
|
||||
<div>
|
||||
<p>{{ title }}</p>
|
||||
<ul>
|
||||
<li v-for="todo in todos" :key="todo.id" @click="increment">
|
||||
{{ todo.id }} - {{ todo.content }}
|
||||
</li>
|
||||
</ul>
|
||||
<p>Count: {{ todoCount }} / {{ meta.totalCount }}</p>
|
||||
<p>Active: {{ active ? 'yes' : 'no' }}</p>
|
||||
<p>Clicks on todos: {{ clickCount }}</p>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script lang="ts">
|
||||
import {
|
||||
defineComponent, PropType, computed, ref, toRef, Ref,
|
||||
} from '@vue/composition-api';
|
||||
import { Todo, Meta } from './models';
|
||||
|
||||
function useClickCount() {
|
||||
const clickCount = ref(0);
|
||||
function increment() {
|
||||
clickCount.value += 1
|
||||
return clickCount.value;
|
||||
}
|
||||
|
||||
return { clickCount, increment };
|
||||
}
|
||||
|
||||
function useDisplayTodo(todos: Ref<Todo[]>) {
|
||||
const todoCount = computed(() => todos.value.length);
|
||||
return { todoCount };
|
||||
}
|
||||
|
||||
export default defineComponent({
|
||||
name: 'CompositionComponent',
|
||||
props: {
|
||||
title: {
|
||||
type: String,
|
||||
required: true
|
||||
},
|
||||
todos: {
|
||||
type: (Array as unknown) as PropType<Todo[]>,
|
||||
default: () => []
|
||||
},
|
||||
meta: {
|
||||
type: (Object as unknown) as PropType<Meta>,
|
||||
required: true
|
||||
},
|
||||
active: {
|
||||
type: Boolean
|
||||
}
|
||||
},
|
||||
setup(props) {
|
||||
return { ...useClickCount(), ...useDisplayTodo(toRef(props, 'todos')) };
|
||||
},
|
||||
});
|
||||
</script>
|
|
@ -1,27 +0,0 @@
|
|||
<template>
|
||||
<div>
|
||||
<essential-link
|
||||
v-for="(link, index) in links"
|
||||
:key="index"
|
||||
:title="link.title"
|
||||
:link="link.link"
|
||||
:icon="link.icon"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script lang="ts">
|
||||
const links = [
|
||||
{ title: 'Neues Home', link: 'newHome', icon: 'mdi-google-home' },
|
||||
{ title: 'Altes Home', link: 'oldHome', icon: 'mdi-home-modern' }
|
||||
];
|
||||
import { defineComponent } from '@vue/composition-api';
|
||||
import EssentialLink from 'components/navigation/EssentialLink.vue';
|
||||
export default defineComponent({
|
||||
// name: 'ComponentName'
|
||||
components: { EssentialLink },
|
||||
setup() {
|
||||
return { links };
|
||||
}
|
||||
});
|
||||
</script>
|
|
@ -0,0 +1,20 @@
|
|||
<template>
|
||||
<q-btn flat dense :icon="icon" :to="{ name: link }" />
|
||||
</template>
|
||||
|
||||
<script lang="ts">
|
||||
import { defineComponent } from '@vue/composition-api';
|
||||
export default defineComponent({
|
||||
name: 'ShortCutLink',
|
||||
props: {
|
||||
link: {
|
||||
required: true,
|
||||
type: String
|
||||
},
|
||||
icon: {
|
||||
required: true,
|
||||
type: String
|
||||
}
|
||||
}
|
||||
});
|
||||
</script>
|
|
@ -23,8 +23,12 @@
|
|||
|
||||
<!-- Hier kommen die Shortlinks hin -->
|
||||
<div>
|
||||
<q-btn flat round dense icon="sim_card" />
|
||||
<q-btn flat round dense icon="gamepad" />
|
||||
<short-cut-link
|
||||
v-for="(shortcut, index) in $flaschengeistPlugins.shortcuts"
|
||||
:key="'shortcut' + index"
|
||||
:link="shortcut.link"
|
||||
:icon="shortcut.icon"
|
||||
/>
|
||||
</div>
|
||||
<q-btn
|
||||
flat
|
||||
|
@ -47,7 +51,7 @@
|
|||
<!-- Plugins -->
|
||||
<q-list>
|
||||
<essential-link
|
||||
v-for="(link, index) in $flaschengeistPluginsMainLinks"
|
||||
v-for="(link, index) in $flaschengeistPlugins.mainLinks"
|
||||
:key="'plugin' + index"
|
||||
:title="link.title"
|
||||
:link="link.link"
|
||||
|
@ -101,8 +105,9 @@
|
|||
|
||||
<script lang="ts">
|
||||
import EssentialLink from 'components/navigation/EssentialLink.vue';
|
||||
import ShortCutLink from 'components/navigation/ShortCutLink.vue';
|
||||
import { Screen } from 'quasar';
|
||||
import { PluginMainLink } from 'boot/plugins';
|
||||
import { LoadedPlugins, PluginMainLink } from 'boot/plugins';
|
||||
import { defineComponent, ref, computed } from '@vue/composition-api';
|
||||
|
||||
const links = [
|
||||
|
@ -114,15 +119,30 @@ const links = [
|
|||
}
|
||||
];
|
||||
|
||||
const shortcuts = [
|
||||
{
|
||||
link: 'about',
|
||||
icon: 'mdi-information'
|
||||
},
|
||||
{
|
||||
link: 'user',
|
||||
icon: 'mdi-account'
|
||||
},
|
||||
{
|
||||
link: 'user-plugin1',
|
||||
icon: 'mdi-account-plus'
|
||||
}
|
||||
];
|
||||
|
||||
declare module 'vue/types/vue' {
|
||||
interface Vue {
|
||||
$flaschengeistPluginsMainLinks: PluginMainLink[];
|
||||
$flaschengeistPlugins: LoadedPlugins;
|
||||
}
|
||||
}
|
||||
|
||||
export default defineComponent({
|
||||
name: 'MainLayout',
|
||||
components: { EssentialLink },
|
||||
components: { EssentialLink, ShortCutLink },
|
||||
setup(_, ctx) {
|
||||
const leftDrawer = ref(false);
|
||||
|
||||
|
@ -142,7 +162,7 @@ export default defineComponent({
|
|||
const pluginChildLinks = computed(() => {
|
||||
const test:
|
||||
| PluginMainLink
|
||||
| undefined = ctx.root.$flaschengeistPluginsMainLinks.find(
|
||||
| undefined = ctx.root.$flaschengeistPlugins.mainLinks.find(
|
||||
(plugin: PluginMainLink) => {
|
||||
if (ctx.root.$route.matched.length > 1) {
|
||||
return plugin.name == ctx.root.$route.matched[1].name;
|
||||
|
@ -170,7 +190,8 @@ export default defineComponent({
|
|||
leftDrawerClicker,
|
||||
links,
|
||||
pluginChildLinks,
|
||||
showRoute
|
||||
showRoute,
|
||||
shortcuts
|
||||
};
|
||||
}
|
||||
});
|
||||
|
|
|
@ -1,49 +0,0 @@
|
|||
<template>
|
||||
<q-page class="row items-center justify-evenly">
|
||||
<example-component
|
||||
title="Example component"
|
||||
active
|
||||
:todos="todos"
|
||||
:meta="meta"
|
||||
></example-component>
|
||||
</q-page>
|
||||
</template>
|
||||
|
||||
<script lang="ts">
|
||||
import { Todo, Meta } from 'components/models';
|
||||
import ExampleComponent from 'components/CompositionComponent.vue';
|
||||
import { defineComponent, ref } from '@vue/composition-api';
|
||||
|
||||
export default defineComponent({
|
||||
name: 'PageIndex',
|
||||
components: { ExampleComponent },
|
||||
setup() {
|
||||
const todos = ref<Todo[]>([
|
||||
{
|
||||
id: 1,
|
||||
content: 'ct1'
|
||||
},
|
||||
{
|
||||
id: 2,
|
||||
content: 'ct2'
|
||||
},
|
||||
{
|
||||
id: 3,
|
||||
content: 'ct3'
|
||||
},
|
||||
{
|
||||
id: 4,
|
||||
content: 'ct4'
|
||||
},
|
||||
{
|
||||
id: 5,
|
||||
content: 'ct5'
|
||||
}
|
||||
]);
|
||||
const meta = ref<Meta>({
|
||||
totalCount: 1200
|
||||
});
|
||||
return { todos, meta };
|
||||
}
|
||||
});
|
||||
</script>
|
|
@ -36,12 +36,12 @@
|
|||
</q-card-section>
|
||||
<q-card-section v-if="$route.name == 'about'">
|
||||
<q-chip
|
||||
v-for="(plugin, index) in $flaschengeistPlugins"
|
||||
v-for="(plugin, index) in $flaschengeistPlugins.plugins"
|
||||
:key="'plugin' + index"
|
||||
>
|
||||
{{ plugin.name }}
|
||||
<q-separator vertical color="black" class="q-ma-xs" />
|
||||
Version
|
||||
{{ plugin.version }}
|
||||
</q-chip>
|
||||
</q-card-section>
|
||||
<q-separator />
|
||||
|
|
|
@ -15,7 +15,7 @@ import { mainLink } from '../plugin';
|
|||
export default defineComponent({
|
||||
// name: 'PageName'
|
||||
setup(_, ctx) {
|
||||
const a = ctx.root.$flaschengeistPluginsMainLinks;
|
||||
const a = ctx.root.$flaschengeistPlugins.mainLinks;
|
||||
return { a, mainLink };
|
||||
}
|
||||
});
|
||||
|
|
|
@ -26,7 +26,10 @@ const plugin: Plugin = {
|
|||
routes,
|
||||
mainLink,
|
||||
name: mainLink.name,
|
||||
requiredModules: []
|
||||
requiredModules: [],
|
||||
shortCuts: [],
|
||||
shortCutsOut: [],
|
||||
version: '1.0.2'
|
||||
};
|
||||
|
||||
export { mainLink };
|
||||
|
|
|
@ -15,7 +15,7 @@ import { mainLink } from '../plugin';
|
|||
export default defineComponent({
|
||||
// name: 'PageName'
|
||||
setup(_, ctx) {
|
||||
const a = ctx.root.$flaschengeistPluginsMainLinks;
|
||||
const a = ctx.root.$flaschengeistPlugins.mainLinks;
|
||||
return { a, mainLink };
|
||||
}
|
||||
});
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import { Plugin, PluginMainLink } from 'boot/plugins';
|
||||
import { Plugin, PluginMainLink, ShortCutLink } from 'boot/plugins';
|
||||
import routes from './routes';
|
||||
|
||||
const mainLink: PluginMainLink = {
|
||||
|
@ -22,11 +22,21 @@ const mainLink: PluginMainLink = {
|
|||
]
|
||||
};
|
||||
|
||||
const shortCuts: ShortCutLink[] = [
|
||||
{
|
||||
link: 'user-plugin2',
|
||||
icon: 'mdi-account-minus'
|
||||
}
|
||||
];
|
||||
|
||||
const plugin: Plugin = {
|
||||
routes,
|
||||
mainLink,
|
||||
name: mainLink.name,
|
||||
requiredModules: ['user']
|
||||
requiredModules: ['user'],
|
||||
shortCuts,
|
||||
shortCutsOut: [],
|
||||
version: '0.1.0'
|
||||
};
|
||||
|
||||
export { mainLink };
|
||||
|
|
|
@ -21,7 +21,7 @@ import { mainLink } from '../plugin';
|
|||
export default defineComponent({
|
||||
// name: 'PageName'
|
||||
setup(_, ctx) {
|
||||
const a = ctx.root.$flaschengeistPluginsMainLinks;
|
||||
const a = ctx.root.$flaschengeistPlugins.mainLinks;
|
||||
return { a, mainLink };
|
||||
}
|
||||
});
|
||||
|
|
|
@ -13,7 +13,10 @@ const plugin: Plugin = {
|
|||
routes,
|
||||
mainLink,
|
||||
name: mainLink.name,
|
||||
requiredModules: []
|
||||
requiredModules: [],
|
||||
shortCutsOut: [],
|
||||
shortCuts: [],
|
||||
version: '0.0.1'
|
||||
};
|
||||
|
||||
export { mainLink };
|
||||
|
|
Loading…
Reference in New Issue