feat: layout/locale/editor/sass等插件兼容vite

This commit is contained in:
wanchun 2022-05-06 15:02:08 +08:00
parent 443879bf4f
commit 15c93eb80e
59 changed files with 1178 additions and 231 deletions

View File

@ -24,7 +24,7 @@ export default (api) => {
api.addRuntimePluginKey(() => 'layout'); api.addRuntimePluginKey(() => 'layout');
const absFilePath = join(namespace, 'index.js'); const absFilePath = join(namespace, 'index.jsx');
const absRuntimeFilePath = join(namespace, 'runtime.js'); const absRuntimeFilePath = join(namespace, 'runtime.js');

View File

@ -1,13 +1,11 @@
import { plugin, ApplyPluginsType } from '@@/core/coreExports'; import { plugin, ApplyPluginsType } from '@@/core/coreExports';
// eslint-disable-next-line import/extensions // eslint-disable-next-line import/extensions
import { access as accessApi } from '../plugin-access/core'; import { access as accessApi } from '../plugin-access/core';
import Exception404 from './views/404'; import Exception404 from './views/404.vue';
import Exception403 from './views/403'; import Exception403 from './views/403.vue';
if (!accessApi) { if (!accessApi) {
throw new Error( throw new Error('[plugin-layout]: pLugin-layout depends on plugin-accessplease install plugin-access first');
'[plugin-layout]: pLugin-layout depends on plugin-accessplease install plugin-access first'
);
} }
const handle = (type, router) => { const handle = (type, router) => {
@ -16,7 +14,7 @@ const handle = (type, router) => {
const name = `Exception${type}`; const name = `Exception${type}`;
const components = { const components = {
404: Exception404, 404: Exception404,
403: Exception403 403: Exception403,
}; };
if (!accesssIds.includes(path)) { if (!accesssIds.includes(path)) {
accessApi.setAccess(accesssIds.concat([path])); accessApi.setAccess(accesssIds.concat([path]));
@ -26,18 +24,19 @@ const handle = (type, router) => {
} }
}; };
export const access = memo => ({ export const access = (memo) => ({
unAccessHandler({ unAccessHandler({ router, to, from, next }) {
router, to, from, next
}) {
const runtimeConfig = plugin.applyPlugins({ const runtimeConfig = plugin.applyPlugins({
key: 'layout', key: 'layout',
type: ApplyPluginsType.modify, type: ApplyPluginsType.modify,
initialValue: {} initialValue: {},
}); });
if (runtimeConfig.unAccessHandler && typeof runtimeConfig.unAccessHandler === 'function') { if (runtimeConfig.unAccessHandler && typeof runtimeConfig.unAccessHandler === 'function') {
return runtimeConfig.unAccessHandler({ return runtimeConfig.unAccessHandler({
router, to, from, next router,
to,
from,
next,
}); });
} }
if (to.path === '/404') { if (to.path === '/404') {
@ -47,17 +46,18 @@ export const access = memo => ({
handle(403, router); handle(403, router);
next('/403'); next('/403');
}, },
noFoundHandler({ noFoundHandler({ router, to, from, next }) {
router, to, from, next
}) {
const runtimeConfig = plugin.applyPlugins({ const runtimeConfig = plugin.applyPlugins({
key: 'layout', key: 'layout',
type: ApplyPluginsType.modify, type: ApplyPluginsType.modify,
initialValue: {} initialValue: {},
}); });
if (runtimeConfig.noFoundHandler && typeof runtimeConfig.noFoundHandler === 'function') { if (runtimeConfig.noFoundHandler && typeof runtimeConfig.noFoundHandler === 'function') {
return runtimeConfig.noFoundHandler({ return runtimeConfig.noFoundHandler({
router, to, from, next router,
to,
from,
next,
}); });
} }
if (to.path === '/403') { if (to.path === '/403') {
@ -67,5 +67,5 @@ export const access = memo => ({
handle(404, router); handle(404, router);
next('/404'); next('/404');
}, },
...memo ...memo,
}); });

View File

@ -12,26 +12,12 @@
> >
<div v-if="routeLayout.logo" class="layout-logo"> <div v-if="routeLayout.logo" class="layout-logo">
<img :src="logo" class="logo-img" /> <img :src="logo" class="logo-img" />
<div class="logo-name">{{title}}</div> <div class="logo-name">{{ title }}</div>
</div> </div>
<Menu <Menu class="layout-menu" :menus="menus" :collapsed="collapsedRef" mode="vertical" :inverted="theme === 'dark'" />
class="layout-menu"
:menus="menus"
:collapsed="collapsedRef"
mode="vertical"
:inverted="theme === 'dark'"
/>
</f-aside> </f-aside>
<f-layout <f-layout :fixed="fixedSideBar" :style="sideStyleRef">
:fixed="fixedSideBar" <f-header v-if="routeLayout.header" ref="headerRef" class="layout-header" :fixed="currentFixedHeaderRef">
:style="sideStyleRef"
>
<f-header
v-if="routeLayout.header"
ref="headerRef"
class="layout-header"
:fixed="currentFixedHeaderRef"
>
<div class="layout-header-custom"> <div class="layout-header-custom">
<slot name="customHeader"></slot> <slot name="customHeader"></slot>
</div> </div>
@ -39,38 +25,23 @@
<slot name="locale"></slot> <slot name="locale"></slot>
</template> </template>
</f-header> </f-header>
<f-layout <f-layout :embedded="!multiTabs" :fixed="currentFixedHeaderRef" :style="headerStyleRef">
:embedded="!multiTabs"
:fixed="currentFixedHeaderRef"
:style="headerStyleRef"
>
<f-main class="layout-main"> <f-main class="layout-main">
<MultiTabProvider :multiTabs="multiTabs" /> <MultiTabProvider :multiTabs="multiTabs" />
</f-main> </f-main>
<f-footer v-if="footer" class="layout-footer"> <f-footer v-if="footer" class="layout-footer">
{{footer}} {{ footer }}
</f-footer> </f-footer>
</f-layout> </f-layout>
</f-layout> </f-layout>
</template> </template>
<template v-if="navigation === 'top'"> <template v-if="navigation === 'top'">
<f-header <f-header v-if="routeLayout.header" ref="headerRef" class="layout-header" :inverted="theme === 'dark'" :fixed="currentFixedHeaderRef">
v-if="routeLayout.header"
ref="headerRef"
class="layout-header"
:inverted="theme === 'dark'"
:fixed="currentFixedHeaderRef"
>
<div v-if="routeLayout.logo" class="layout-logo"> <div v-if="routeLayout.logo" class="layout-logo">
<img :src="logo" class="logo-img" /> <img :src="logo" class="logo-img" />
<div class="logo-name">{{title}}</div> <div class="logo-name">{{ title }}</div>
</div> </div>
<Menu <Menu class="layout-menu" :menus="menus" mode="horizontal" :inverted="theme === 'dark'" />
class="layout-menu"
:menus="menus"
mode="horizontal"
:inverted="theme === 'dark'"
/>
<div class="layout-header-custom"> <div class="layout-header-custom">
<slot name="customHeader"></slot> <slot name="customHeader"></slot>
</div> </div>
@ -78,30 +49,20 @@
<slot name="locale"></slot> <slot name="locale"></slot>
</template> </template>
</f-header> </f-header>
<f-layout <f-layout :embedded="!multiTabs" :fixed="currentFixedHeaderRef" :style="headerStyleRef">
:embedded="!multiTabs"
:fixed="currentFixedHeaderRef"
:style="headerStyleRef"
>
<f-main class="layout-main"> <f-main class="layout-main">
<MultiTabProvider :multiTabs="multiTabs" /> <MultiTabProvider :multiTabs="multiTabs" />
</f-main> </f-main>
<f-footer v-if="footer" class="layout-footer"> <f-footer v-if="footer" class="layout-footer">
{{footer}} {{ footer }}
</f-footer> </f-footer>
</f-layout> </f-layout>
</template> </template>
<template v-if="navigation === 'mixin'"> <template v-if="navigation === 'mixin'">
<f-header <f-header v-if="routeLayout.header" ref="headerRef" class="layout-header" :fixed="currentFixedHeaderRef" :inverted="theme === 'dark'">
v-if="routeLayout.header"
ref="headerRef"
class="layout-header"
:fixed="currentFixedHeaderRef"
:inverted="theme === 'dark'"
>
<div v-if="routeLayout.logo" class="layout-logo"> <div v-if="routeLayout.logo" class="layout-logo">
<img :src="logo" class="logo-img" /> <img :src="logo" class="logo-img" />
<div class="logo-name">{{title}}</div> <div class="logo-name">{{ title }}</div>
</div> </div>
<div class="layout-header-custom"> <div class="layout-header-custom">
<slot name="customHeader"></slot> <slot name="customHeader"></slot>
@ -119,23 +80,14 @@
collapsible collapsible
class="layout-aside" class="layout-aside"
> >
<Menu <Menu class="layout-menu" :menus="menus" :collapsed="collapsedRef" mode="vertical" />
class="layout-menu"
:menus="menus"
:collapsed="collapsedRef"
mode="vertical"
/>
</f-aside> </f-aside>
<f-layout <f-layout :embedded="!multiTabs" :fixed="fixedSideBar" :style="sideStyleRef">
:embedded="!multiTabs"
:fixed="fixedSideBar"
:style="sideStyleRef"
>
<f-main class="layout-main"> <f-main class="layout-main">
<MultiTabProvider :multiTabs="multiTabs" /> <MultiTabProvider :multiTabs="multiTabs" />
</f-main> </f-main>
<f-footer v-if="footer" class="layout-footer"> <f-footer v-if="footer" class="layout-footer">
{{footer}} {{ footer }}
</f-footer> </f-footer>
</f-layout> </f-layout>
</f-layout> </f-layout>
@ -147,11 +99,9 @@
<script> <script>
import { ref, computed, onMounted } from 'vue'; import { ref, computed, onMounted } from 'vue';
import { useRoute, plugin, ApplyPluginsType } from '@@/core/coreExports'; import { useRoute, plugin, ApplyPluginsType } from '@@/core/coreExports';
import { import { FLayout, FAside, FMain, FFooter, FHeader } from '@fesjs/fes-design';
FLayout, FAside, FMain, FFooter, FHeader import Menu from './Menu.vue';
} from '@fesjs/fes-design'; import MultiTabProvider from './MultiTabProvider.vue';
import Menu from './Menu';
import MultiTabProvider from './MultiTabProvider';
import defaultLogo from '../assets/logo.png'; import defaultLogo from '../assets/logo.png';
export default { export default {
@ -162,52 +112,52 @@ export default {
FFooter, FFooter,
FHeader, FHeader,
Menu, Menu,
MultiTabProvider MultiTabProvider,
}, },
props: { props: {
menus: { menus: {
type: Array, type: Array,
default() { default() {
return []; return [];
} },
}, },
title: { title: {
type: String, type: String,
default: '' default: '',
}, },
locale: { locale: {
type: Boolean, type: Boolean,
default: false default: false,
}, },
logo: { logo: {
type: String, type: String,
default: defaultLogo default: defaultLogo,
}, },
theme: { theme: {
type: String, type: String,
default: 'dark' // lightdark default: 'dark', // lightdark
}, },
navigation: { navigation: {
type: String, type: String,
default: 'side' // side / top / mixin // default: 'side', // side / top / mixin //
}, },
fixedHeader: { fixedHeader: {
type: Boolean, type: Boolean,
default: false default: false,
}, },
fixedSideBar: { fixedSideBar: {
type: Boolean, type: Boolean,
default: true default: true,
}, },
multiTabs: { multiTabs: {
type: Boolean, type: Boolean,
default: false default: false,
}, },
sideWidth: { sideWidth: {
type: Number, type: Number,
default: 200 default: 200,
}, },
footer: String footer: String,
}, },
setup(props) { setup(props) {
const headerRef = ref(); const headerRef = ref();
@ -227,8 +177,8 @@ export default {
initialValue: { initialValue: {
sidebar: true, sidebar: true,
header: true, header: true,
logo: true logo: true,
} },
}); });
const routeLayout = computed(() => { const routeLayout = computed(() => {
let config; let config;
@ -239,9 +189,7 @@ export default {
} else if (typeof metaLayoutConfig === 'object') { } else if (typeof metaLayoutConfig === 'object') {
config = { ...runtimeConfig, ...metaLayoutConfig }; config = { ...runtimeConfig, ...metaLayoutConfig };
} else { } else {
console.error( console.error('[plugin-layout]: meta layout must be object or boolean');
'[plugin-layout]: meta layout must be object or boolean'
);
} }
// query layout false // query layout false
const routeQueryLayoutConfig = route.query.layout && JSON.parse(route.query.layout); const routeQueryLayoutConfig = route.query.layout && JSON.parse(route.query.layout);
@ -250,21 +198,19 @@ export default {
} else if (typeof routeQueryLayoutConfig === 'object') { } else if (typeof routeQueryLayoutConfig === 'object') {
config = { ...config, ...routeQueryLayoutConfig }; config = { ...config, ...routeQueryLayoutConfig };
} else if (routeQueryLayoutConfig !== undefined) { } else if (routeQueryLayoutConfig !== undefined) {
console.error( console.error('[plugin-layout]: query layout must be object or boolean');
'[plugin-layout]: query layout must be object or boolean'
);
} }
return config; return config;
}); });
const currentFixedHeaderRef = computed( const currentFixedHeaderRef = computed(() => props.fixedHeader || props.navigation === 'mixin');
() => props.fixedHeader || props.navigation === 'mixin'
);
const headerStyleRef = computed(() => (currentFixedHeaderRef.value ? { top: `${headerHeightRef.value}px` } : null)); const headerStyleRef = computed(() => (currentFixedHeaderRef.value ? { top: `${headerHeightRef.value}px` } : null));
const sideStyleRef = computed(() => (props.fixedSideBar const sideStyleRef = computed(() =>
? { props.fixedSideBar
left: collapsedRef.value ? '48px' : `${props.sideWidth}px` ? {
} left: collapsedRef.value ? '48px' : `${props.sideWidth}px`,
: null)); }
: null,
);
return { return {
headerRef, headerRef,
headerHeightRef, headerHeightRef,
@ -273,9 +219,9 @@ export default {
collapsedRef, collapsedRef,
currentFixedHeaderRef, currentFixedHeaderRef,
headerStyleRef, headerStyleRef,
sideStyleRef sideStyleRef,
}; };
} },
}; };
</script> </script>
<style lang="less" scoped> <style lang="less" scoped>

View File

@ -1,61 +1,57 @@
<template> <template>
<f-menu <f-menu :modelValue="activePath" :inverted="inverted" :mode="mode" :options="fixedMenus" @select="onMenuClick"></f-menu>
:modelValue="activePath"
:inverted="inverted"
:mode="mode"
:options="fixedMenus"
@select="onMenuClick"
></f-menu>
</template> </template>
<script> <script>
import { computed, h } from 'vue'; import { computed, h } from 'vue';
import { FMenu } from '@fesjs/fes-design'; import { FMenu } from '@fesjs/fes-design';
import { useRoute, useRouter } from '@@/core/coreExports'; import { useRoute, useRouter } from '@@/core/coreExports';
import MenuIcon from './MenuIcon'; import MenuIcon from './MenuIcon.vue';
import { transform as transformByAccess } from '../helpers/pluginAccess'; import { transform as transformByAccess } from '../helpers/pluginAccess';
import { transform as transformByLocale } from '../helpers/pluginLocale'; import { transform as transformByLocale } from '../helpers/pluginLocale';
import { flatNodes } from '../helpers/utils'; import { flatNodes } from '../helpers/utils';
export default { export default {
components: { components: {
FMenu FMenu,
}, },
props: { props: {
menus: { menus: {
type: Array, type: Array,
default() { default() {
return []; return [];
} },
}, },
mode: { mode: {
type: String, type: String,
default: 'vertical' default: 'vertical',
}, },
inverted: { inverted: {
type: Boolean, type: Boolean,
default: false default: false,
} },
}, },
setup(props) { setup(props) {
const route = useRoute(); const route = useRoute();
const router = useRouter(); const router = useRouter();
const transform = menus => menus.map((menu) => { const transform = (menus) =>
const copy = { menus.map((menu) => {
...menu, const copy = {
label: menu.title, ...menu,
value: menu.path label: menu.title,
}; value: menu.path,
if (menu.icon) { };
copy.icon = () => h(MenuIcon, { if (menu.icon) {
icon: menu.icon copy.icon = () =>
}); h(MenuIcon, {
} icon: menu.icon,
if (menu.children) { });
copy.children = transform(menu.children); }
} if (menu.children) {
return copy; copy.children = transform(menu.children);
}); }
return copy;
});
const fixedMenus = computed(() => transformByLocale(transformByAccess(transform(props.menus)))); const fixedMenus = computed(() => transformByLocale(transformByAccess(transform(props.menus))));
const menus = computed(() => flatNodes(fixedMenus.value)); const menus = computed(() => flatNodes(fixedMenus.value));
const activePath = computed(() => { const activePath = computed(() => {
@ -81,16 +77,14 @@ export default {
} else if (/^\//.test(path)) { } else if (/^\//.test(path)) {
router.push(path); router.push(path);
} else { } else {
console.warn( console.warn('[plugin-layout]: 菜单的path只能使以http(s)开头的网址或者路由地址');
'[plugin-layout]: 菜单的path只能使以http(s)开头的网址或者路由地址'
);
} }
}; };
return { return {
activePath, activePath,
fixedMenus, fixedMenus,
onMenuClick onMenuClick,
}; };
} },
}; };
</script> </script>

View File

@ -1,4 +1,4 @@
<script> <script lang="jsx">
import { ref, onBeforeMount } from 'vue'; import { ref, onBeforeMount } from 'vue';
// eslint-disable-next-line import/extensions // eslint-disable-next-line import/extensions
import Icons from '../icons'; import Icons from '../icons';
@ -6,7 +6,7 @@ import { validateContent } from '../helpers/svg';
export default { export default {
props: { props: {
icon: [String, Object] icon: [String, Object],
}, },
setup(props) { setup(props) {
const AIcon = ref(null); const AIcon = ref(null);
@ -31,16 +31,11 @@ export default {
return <AIcon.value />; return <AIcon.value />;
} }
if (AText.value) { if (AText.value) {
return ( return <span class={'fes-layout-icon'} innerHTML={AText.value}></span>;
<span
class={'fes-layout-icon'}
innerHTML={AText.value}
></span>
);
} }
return null; return null;
}; };
} },
}; };
</script> </script>
<style> <style>

View File

@ -1,6 +1,5 @@
import { readFileSync } from 'fs'; import { readFileSync } from 'fs';
import { join } from 'path'; import { join } from 'path';
import { getLocalesJSON } from './utils';
import { name } from '../package.json'; import { name } from '../package.json';
const namespace = 'plugin-locale'; const namespace = 'plugin-locale';
@ -21,6 +20,8 @@ export default (api) => {
}, },
}); });
const { getLocales } = require('./utils');
api.addRuntimePluginKey(() => 'locale'); api.addRuntimePluginKey(() => 'locale');
const absoluteFilePath = join(namespace, 'core.js'); const absoluteFilePath = join(namespace, 'core.js');
@ -46,7 +47,7 @@ export default (api) => {
const localeConfigFileBasePath = getLocaleFileBasePath(); const localeConfigFileBasePath = getLocaleFileBasePath();
const locales = getLocalesJSON(localeConfigFileBasePath); const locales = getLocales(localeConfigFileBasePath);
api.writeTmpFile({ api.writeTmpFile({
path: absoluteFilePath, path: absoluteFilePath,

View File

@ -8,12 +8,22 @@
import { isRef, unref } from 'vue'; import { isRef, unref } from 'vue';
import { createI18n, useI18n } from '{{{ VUE_I18N_PATH }}}'; import { createI18n, useI18n } from '{{{ VUE_I18N_PATH }}}';
import { plugin, ApplyPluginsType } from "@@/core/coreExports"; import { plugin, ApplyPluginsType } from "@@/core/coreExports";
import SelectLang from "./views/SelectLang"; import SelectLang from "./views/SelectLang.vue";
{{#REPLACE_LOCALES}}
import {{importName}} from "{{{path}}}";
{{/REPLACE_LOCALES}}
// 共享出去 // 共享出去
plugin.share("locale", {useI18n, SelectLang }); plugin.share("locale", {useI18n, SelectLang });
const locales = {{{REPLACE_LOCALES}}}; const locales = [
{{#REPLACE_LOCALES}}
{
locale: "{{locale}}",
message: {{importName}}
},
{{/REPLACE_LOCALES}}
];
const defaultOptions = {{{REPLACE_DEFAULT_OPTIONS}}}; const defaultOptions = {{{REPLACE_DEFAULT_OPTIONS}}};

View File

@ -4,30 +4,18 @@ import { join, basename } from 'path';
export function getLocales(cwd) { export function getLocales(cwd) {
const files = glob const files = glob
.sync('*.js', { .sync('*.js', {
cwd cwd,
}) })
.filter( .filter((file) => !file.endsWith('.d.ts') && !file.endsWith('.test.js') && !file.endsWith('.test.jsx'))
file => !file.endsWith('.d.ts') .map((fileName) => {
&& !file.endsWith('.test.js')
&& !file.endsWith('.test.jsx')
).map((fileName) => {
const locale = basename(fileName, '.js'); const locale = basename(fileName, '.js');
const importName = locale.replace('-', '');
return { return {
importName,
locale, locale,
message: `require('${join(cwd, fileName)}').default` path: join(cwd, fileName),
}; };
}); });
return files; return files;
} }
export function getLocalesJSON(cwd) {
const locales = getLocales(cwd);
return JSON.stringify(locales, null, 2)
.replace(
/"message": ("(.+?)")/g,
(global, m1, m2) => `"message": ${m2.replace(/\^/g, '"')}`
)
.replace(/\\r\\n/g, '\r\n')
.replace(/\\n/g, '\r\n');
}

View File

@ -28,7 +28,8 @@
"@fesjs/utils": "^2.0.4", "@fesjs/utils": "^2.0.4",
"lodash-es": "^4.17.15", "lodash-es": "^4.17.15",
"monaco-editor": "^0.20.0", "monaco-editor": "^0.20.0",
"monaco-editor-webpack-plugin": "^1.9.1" "monaco-editor-webpack-plugin": "^1.9.1",
"vite-plugin-monaco-editor": "^1.0.10"
}, },
"peerDependencies": { "peerDependencies": {
"@fesjs/fes": "^2.0.0", "@fesjs/fes": "^2.0.0",

View File

@ -26,9 +26,7 @@ export default (api) => {
}); });
const absoluteFilePath = join(namespace, 'core.js'); const absoluteFilePath = join(namespace, 'core.js');
const absRuntimeFilePath = join(namespace, 'runtime.js'); const absRuntimeFilePath = join(namespace, 'runtime.js');
const absLoaderFilePath = join(namespace, 'loader.js'); const absLoaderFilePath = join(namespace, 'loader.js');
const absEditorFilePath = join(namespace, 'editor.vue'); const absEditorFilePath = join(namespace, 'editor.vue');
@ -76,10 +74,18 @@ export default (api) => {
api.addRuntimePlugin(() => `@@/${absRuntimeFilePath}`); api.addRuntimePlugin(() => `@@/${absRuntimeFilePath}`);
api.chainWebpack((webpackConfig) => { if (api.builder.isVite) {
webpackConfig.plugin('monaco-editor').use(require('monaco-editor-webpack-plugin'), [api.config?.monacoEditor || {}]); api.modifyBundleConfig((config) => {
return webpackConfig; const monacoEditorPlugin = require('vite-plugin-monaco-editor').default;
}); config?.plugins?.push(monacoEditorPlugin(api.config?.monacoEditor || {}));
});
//
} else {
api.chainWebpack((webpackConfig) => {
webpackConfig.plugin('monaco-editor').use(require('monaco-editor-webpack-plugin'), [api.config?.monacoEditor || {}]);
return webpackConfig;
});
}
api.addConfigType(() => ({ api.addConfigType(() => ({
source: name, source: name,

View File

@ -1,4 +1,4 @@
import Editor from './editor'; import Editor from './editor.vue';
import _monaco from './loader'; import _monaco from './loader';
export const MonacoEditor = Editor; export const MonacoEditor = Editor;

View File

@ -1,8 +1,5 @@
export default (api) => { export default (api) => {
const { const { utils } = api;
utils
} = api;
api.describe({ api.describe({
key: 'sass', key: 'sass',
@ -13,25 +10,29 @@ export default (api) => {
sassOptions: joi.object(), sassOptions: joi.object(),
prependData: joi.alternatives(joi.string(), joi.func()), prependData: joi.alternatives(joi.string(), joi.func()),
sourceMap: joi.boolean(), sourceMap: joi.boolean(),
webpackImporter: joi.boolean() webpackImporter: joi.boolean(),
}); });
}, },
default: {} default: {},
} },
}); });
api.chainWebpack((memo, { createCSSRule }) => { if (api.builder.isVite) {
createCSSRule({ // vite 不需要处理
lang: 'sass', } else {
test: /\.(sass|scss)(\?.*)?$/, api.chainWebpack((memo, { createCSSRule }) => {
loader: require.resolve('sass-loader'), createCSSRule({
options: utils.deepmerge( lang: 'sass',
{ test: /\.(sass|scss)(\?.*)?$/,
implementation: require('sass') loader: require.resolve('sass-loader'),
}, options: utils.deepmerge(
api.config.sass || {} {
) implementation: require('sass'),
},
api.config.sass || {},
),
});
return memo;
}); });
return memo; }
});
}; };

View File

@ -13,5 +13,5 @@ export default (api) => {
.filter((file) => existsSync(file)) .filter((file) => existsSync(file))
.slice(0, 1); .slice(0, 1);
api.addEntryCodeAhead(() => `${globalCSSFile.map((file) => `require('${winPath(relative(absTmpPath, file))}');`).join('')}`); api.addEntryCodeAhead(() => `${globalCSSFile.map((file) => `import '${winPath(relative(absTmpPath, file))}';`).join('')}`);
}; };

View File

@ -0,0 +1 @@
FES_APP_PUBLISH_ERROR_PAGE=helloworld

View File

@ -0,0 +1 @@
FES_APP_PUBLISH_ERROR_PAGE=helloProduction

View File

@ -0,0 +1,11 @@
module.exports = {
extends: ['@webank/eslint-config-webank/vue.js'],
overrides: [
{
files: ['**/__tests__/*.{j,t}s?(x)', '**/tests/unit/**/*.spec.{j,t}s?(x)'],
},
],
env: {
jest: true,
},
};

View File

@ -0,0 +1,90 @@
// .fes.js 只负责管理编译时配置只能使用plain Object
export default {
// exportStatic: {},
define: {
__DEV__: false
},
title: '海贼王',
router: {
mode: 'hash'
},
access: {
roles: {
admin: ['*'],
menuTest: ['/', '/menuTest']
}
},
request: {
dataField: 'result'
},
mock: {
prefix: '/v2'
},
proxy: {
'/v2': {
target: 'https://api.douban.com/',
changeOrigin: true
}
},
layout: {
title: 'Fes.js',
footer: 'Created by MumbleFE',
multiTabs: false,
navigation: 'side',
theme: 'dark',
menus: [
{
name: 'index',
icon: '/wine-outline.svg',
match: ['/route/*']
},
{
name: 'store'
},
{
name: 'editor',
icon: '/wine-outline.svg'
},
{
title: '$externalLink',
icon: 'UserOutlined',
path: 'https://www.baidu.com'
},
{
name: 'mock'
},
{
title: '菜单权限测试',
children: [
{
title: '子菜单',
path: '/menuTest'
},
]
},
{
name: 'cssModule'
},
{
name: 'pinia'
}
]
},
locale: {
legacy: true
},
enums: {
status: [
['0', '无效的'],
['1', '有效的']
]
},
vuex: {
strict: true
},
dynamicImport: true,
monacoEditor: {
languages: ['javascript', 'typescript', 'html', 'json']
}
};

View File

@ -0,0 +1,5 @@
export default {
// define: {
// __DEV__: true
// },
}

View File

@ -0,0 +1,6 @@
// .fes.js 只负责管理编译时配置只能使用plain Object
export default {
publicPath: 'https://gw.alipayobjects.com/'
};

11
packages/fes-template-vite1/.gitignore vendored Normal file
View File

@ -0,0 +1,11 @@
.DS_Store
# dependencies
/node_modules
/coverage
# fes
/src/.fes
/src/.fes-production
/src/.fes-test
/.env.local

View File

@ -0,0 +1,21 @@
MIT License
Copyright (c) 2020-present webank
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.

View File

@ -0,0 +1,12 @@
# fes 模版
内部测试用,不对外发布
## 环境变量
* 业务代码使用的全局变量,使用 webpack define 定义
* 针对不同的环境构建的变量
* 开发环境 .evn.local
* .env 定义环境变量
* .env.xxx 定义特定的环境变

View File

@ -0,0 +1,5 @@
import sum from '@/utils/sum';
test('adds 1 + 2 to equal 3', () => {
expect(sum(1, 2)).toBe(3);
});

View File

@ -0,0 +1,20 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>
<%= title %>
</title>
<link rel="shortcut icon" type="image/x-icon" href="./logo.png">
</head>
<body>
<div id="<%= mountElementId %>"></div>
<script>
console.log('<%= FES_APP_PUBLISH_ERROR_PAGE %>');
</script>
</body>
</html>

View File

@ -0,0 +1,143 @@
export default function ({ cgiMock, mockjs, utils }) {
const { Random } = mockjs;
// 测试 proxy 与 mock 用例集合
cgiMock('/movie/in_theaters_mock', (req, res) => {
res.send(
JSON.stringify({
code: '0',
msg: '',
result: {
text: 'movie: movie/in_theaters_mock ~~~~~',
},
}),
);
});
cgiMock('/movie/test_mock', (req, res) => {
res.send(
JSON.stringify({
code: '0',
msg: '',
result: {
text: 'mock: movie/test_mock',
},
}),
);
});
// 测试用例: mock.js change重现请求需要能拉最新的数据
cgiMock('/watchtest', (req, res) => {
res.send(
JSON.stringify({
code: '0',
msg: '',
result: {
text: '通过 register 测试 mock watch: 初始状态',
},
}),
);
});
// 返回一个数字
// cgiMock('/number', 666);
cgiMock('/number', 999);
// 返回一个json
cgiMock({
url: '/json',
result: {
code: '400101',
msg: "不合法的请求:Missing cookie 'wb_app_id' for method parameter of type String",
transactionTime: '20170309171146',
success: false,
},
});
// 利用 mock.js 产生随机文本
cgiMock('/text', Random.cparagraph());
// 返回一个字符串 利用 mock.js 产生随机字符
cgiMock(
'/random',
mockjs.mock({
'string|1-10': '★',
}),
);
// 正则匹配url, 返回一个字符串
cgiMock(/\/abc|\/xyz/, 'regexp test!');
// option.result 参数如果是一个函数, 可以实现自定义返回内容, 接收的参数是是经过 express 封装的 req 和 res 对象.
cgiMock(/\/function$/, (req, res) => {
res.send('function test');
});
// 返回文本 readFileSync
cgiMock('/file', utils.file('./package.json'));
// 更复杂的规则配置
cgiMock({
url: /\/who/,
method: 'GET',
result(req, res) {
if (req.query.name === 'kwan') {
res.json({ kwan: '孤独患者' });
} else {
res.send('Nooooooooooo');
}
},
headers: {
'Content-Type': 'text/plain',
'Content-Length': '123',
ETag: '12345',
},
cookies: [
{
name: 'myname',
value: 'kwan',
maxAge: 900000,
httpOnly: true,
},
],
});
// 携带参数的请求
cgiMock('/v2/audit/list', (req, res) => {
const { currentPage, pageSize, isAudited } = req.body;
res.send({
code: '0',
msg: '',
data: {
currentPage,
pageSize,
totalPage: 2,
totalCount: 12,
pageData: Array.from({ length: pageSize }, () => ({
title: Random.title(),
authorName: Random.cname(),
authorId: Random.name(),
createTime: Date.now(),
updateTime: Date.now(),
readCount: Random.integer(60, 1000),
favoriteCount: Random.integer(1, 50),
postId: '12323',
serviceTag: '业务类型',
productTag: '产品类型',
requestTag: '需求类型',
handleTag: '已采纳',
postType: 'voice',
postStatus: isAudited ? 'pass' : 'auditing',
auditStatus: 'audit1',
})),
},
});
});
// multipart/form-data 类型
cgiMock('/v2/upload', (req, res) => {
res.send({
code: '0',
msg: '文件上传成功',
});
});
}

View File

@ -0,0 +1,66 @@
{
"name": "@fesjs/template",
"version": "2.0.0",
"description": "fes项目模版",
"scripts": {
"build": "fes build",
"prod": "FES_ENV=prod fes build",
"analyze": "ANALYZE=1 fes build",
"dev": "fes dev",
"test": "fes test"
},
"keywords": [
"管理端",
"fes",
"fast",
"easy",
"strong"
],
"files": [
".eslintrc.js",
".gitignore",
".fes.js",
".fes.prod.js",
"mock.js",
"package.json",
"README.md",
"tsconfig.json",
"/src",
"/config"
],
"repository": {
"type": "git",
"url": "git+https://github.com/WeBankFinTech/fes.js.git",
"directory": "packages/fes-template"
},
"author": "harrywan",
"license": "MIT",
"bugs": {
"url": "https://github.com/WeBankFinTech/fes.js/issues"
},
"homepage": "https://github.com/WeBankFinTech/fes.js#readme",
"publishConfig": {
"access": "public"
},
"dependencies": {
"@fesjs/fes": "^2.0.0",
"@fesjs/plugin-access": "^2.0.0",
"@fesjs/plugin-layout": "^4.0.0",
"@fesjs/plugin-locale": "^3.0.0",
"@fesjs/plugin-model": "^2.0.0",
"@fesjs/plugin-enums": "^2.0.0",
"@fesjs/plugin-jest": "^2.0.0",
"@fesjs/plugin-vuex": "^2.0.0",
"@fesjs/plugin-request": "^2.0.0",
"@fesjs/plugin-sass": "^2.0.0",
"@fesjs/plugin-monaco-editor": "^2.0.0-beta.0",
"@fesjs/plugin-windicss": "^2.0.0",
"@fesjs/plugin-pinia": "^2.0.0",
"@fesjs/fes-design": "^0.3.3",
"@fesjs/build-vite": "^1.0.0",
"vue": "^3.0.5",
"vuex": "^4.0.0",
"pinia": "^2.0.11"
},
"private": true
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 33 KiB

View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" class="ionicon" viewBox="0 0 512 512"><title>Wine</title><path d="M398.57 80H113.43v16S87.51 272 256 272 398.57 96 398.57 96zM256 272v160" fill="none" stroke="currentColor" stroke-linejoin="round" stroke-width="32"/><path fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round" stroke-width="32" d="M352 432H160"/><path fill="none" stroke="currentColor" stroke-linejoin="round" stroke-width="32" d="M112 160h288"/></svg>

After

Width:  |  Height:  |  Size: 485 B

View File

@ -0,0 +1,27 @@
import { access as accessApi, pinia } from '@fesjs/fes';
import PageLoading from '@/components/PageLoading.vue';
import UserCenter from '@/components/UserCenter.vue';
import { useStore } from '@/store/main';
export const beforeRender = {
loading: <PageLoading />,
action() {
const { setRole } = accessApi;
return new Promise((resolve) => {
setTimeout(() => {
const store = useStore(pinia);
store.$patch({
userName: '李雷',
});
setRole('admin');
resolve({
userName: '李雷',
});
}, 1000);
});
},
};
export const layout = {
customHeader: <UserCenter />,
};

View File

@ -0,0 +1,19 @@
import { requestWrap } from '@fesjs/fes';
// TODO
// 响应体控制
// formData 控制
// 错误控制
// 跳错误页面 || 或者重新登录
// 段时间内不能重复发送的请求
// or
export default requestWrap({
login: {
url: '',
throttle: 300,
options: {
method: 'get'
}
}
});

View File

@ -0,0 +1,30 @@
<template>
<div class="page-loading">
<f-spin size="large" stroke="#5384ff" />
</div>
</template>
<script>
import { FSpin } from '@fesjs/fes-design';
export default {
components: {
FSpin
},
setup() {
return {
};
}
};
</script>
<style>
.page-loading{
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
display: flex;
align-items: center;
justify-content: center;
}
</style>

View File

@ -0,0 +1,20 @@
<template>
<div class="user-center">{{initialState.userName}}</div>
</template>
<script>
import { useModel } from '@fesjs/fes';
export default {
setup() {
const initialState = useModel('@@initialState');
return {
initialState
};
}
};
</script>
<style lang="less">
.user-center {
text-align: right;
}
</style>

View File

@ -0,0 +1,5 @@
html {
body {
font-size: 16px;
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.5 KiB

View File

@ -0,0 +1,8 @@
export default {
home: 'home',
store: 'store',
editor: 'editor',
externalLink: 'externalLink',
mock: 'mock'
};

View File

@ -0,0 +1,8 @@
export default {
home: '首页',
store: '状态管理',
editor: '编辑器',
externalLink: '外部链接',
mock: '代理'
};

View File

@ -0,0 +1,8 @@
import { ref } from 'vue';
export default function user() {
const count = ref(1);
return {
count
};
}

View File

@ -0,0 +1,24 @@
<template>
<div :class="$style.red">
字体颜色
</div>
</template>
<config>
{
"name": "cssModule",
"title": "css Module 测试"
}
</config>
<script>
export default {
setup() {
return {};
}
};
</script>
<style module>
.red {
color: red;
}
</style>

View File

@ -0,0 +1,42 @@
<template>
<monaco-editor
ref="editorRef"
v-model="json"
:language="language"
height="200px"
check
/>
{{json}}
</template>
<config>
{
"name": "editor",
"title": "$editor",
"keep-alive": true
}
</config>
<script>
import { onMounted, ref } from 'vue';
import { MonacoEditor } from '@fesjs/fes';
export default {
components: {
MonacoEditor
},
setup() {
const editorRef = ref();
const json = ref('');
const language = ref('json');
onMounted(() => {
setTimeout(() => {
language.value = 'html';
}, 3000);
});
return {
editorRef,
json,
language
};
}
};
</script>

View File

@ -0,0 +1,32 @@
<template>
<div class="page">
home222
<FButton class="m-2">Button</FButton>
</div>
</template>
<script>
import { FButton } from '@fesjs/fes-design';
export default {
components: {
FButton,
},
setup() {
return {};
},
};
</script>
<style>
.page {
height: 1000px;
}
</style>
<config>
{
"name": "index",
"title": "$home"
}
</config>

View File

@ -0,0 +1,30 @@
<template>
<div class="page">
menuTest: {{route.params}}
</div>
</template>
<config>
{
"title": "menuTest-详情"
}
</config>
<script>
import { useRoute } from '@fesjs/fes';
export default {
components: {
},
setup() {
const route = useRoute();
return {
route
};
}
};
</script>
<style>
.page {
min-height: 100vh;
}
</style>

View File

@ -0,0 +1,26 @@
<template>
<div class="page">
menuTest-index
</div>
</template>
<config>
{
"title": "menuTest"
}
</config>
<script>
export default {
components: {
},
setup() {
return {
};
}
};
</script>
<style>
.page {
min-height: 100vh;
}
</style>

View File

@ -0,0 +1,8 @@
<template>
<div style="display: flex;flex-direction: column;">
<router-link to="/menuTest/1">Go to 1</router-link>
<router-link to="/menuTest/2">Go to 2</router-link>
<router-link to="/menuTest/3">Go to 3</router-link>
</div>
<router-view />
</template>

View File

@ -0,0 +1,47 @@
<template>
<div>mock and proxy</div>
</template>
<config>
{
"name": "mock",
"title": "$mock"
}
</config>
<script setup>
import { request } from '@fesjs/fes';
console.log('测试 mock!!');
request('/v2/file')
.then((data) => {
console.log(data);
})
.catch((err) => {
console.log(err);
});
request('/v2/movie/in_theaters_mock', { a: 1 }, 'get')
.then((data) => {
console.log(data);
})
.catch((err) => {
console.log(err);
});
console.log('测试 proxy!!');
request(
'/v2/movie/in_theaters_proxy',
{ a: 1 },
{
method: 'get',
headers: { Accept: '*/*' },
},
)
.then((resp) => {
console.log(resp);
})
.catch((err) => {
console.log(err);
});
</script>

View File

@ -0,0 +1,31 @@
<template>
<div>{{store.counter}}</div>
<FButton class="m-2" @click="store.increment">Button</FButton>
</template>
<config>
{
"name": "pinia",
"title": "pinia"
}
</config>
<script>
import { useStore } from '@/store/main';
import { FButton } from '@fesjs/fes-design';
export default {
components: {
FButton
},
setup() {
const store = useStore();
console.log(store);
return {
store
};
}
};
</script>
<style>
</style>

View File

@ -0,0 +1,23 @@
<template>
<div>
{{params.id}}
</div>
</template>
<config>
{
"name": "activeRoute",
"title": "动态路由"
}
</config>
<script>
import { useRoute } from '@fesjs/fes';
export default {
setup() {
const { params } = useRoute();
return {
params
};
}
};
</script>

View File

@ -0,0 +1,60 @@
<template>
<div class="page">
<h4>Vuex</h4>
<div>
<button @click="increment">click me{{doubleCount}}</button>
</div>
<div>
<button :disabled="disabled" @click="login">async login</button>
</div>
<div>
<button @click="fooBarIncrement">
foo/bar{{fooBarDoubleCount}}
</button>
</div>
<div>{{address}}</div>
</div>
</template>
<config>
{
"name": "store",
"title": "$store"
}
</config>
<script>
import { computed, ref } from 'vue';
import { useStore } from 'vuex';
import { MUTATION_TYPES, GETTER_TYPES, ACTION_TYPES } from '@fesjs/fes';
export default {
setup() {
const store = useStore();
console.log('store==>', store);
const disabled = ref(false);
return {
address: computed(() => store.getters[GETTER_TYPES.user.address]),
doubleCount: computed(
() => store.getters[GETTER_TYPES.counter.doubleCount]
),
disabled,
increment: () => store.commit(MUTATION_TYPES.counter.increment),
login: () => {
disabled.value = true;
store.dispatch(ACTION_TYPES.user.login).then((res) => {
// eslint-disable-next-line no-alert
window.alert(res);
disabled.value = false;
});
},
fooBarIncrement: () => store.commit(MUTATION_TYPES.fooBar.increment),
fooBarDoubleCount: computed(
() => store.getters[GETTER_TYPES.fooBar.doubleCount]
)
};
}
};
</script>
<style scoped>
.page {
}
</style>

View File

@ -0,0 +1,21 @@
import { defineStore } from 'pinia';
// useStore could be anything like useUser, useCart
// the first argument is a unique id of the store across your application
export const useStore = defineStore('main', {
// other options...
state: () => ({
// all these properties will have their type inferred automatically
counter: 0,
name: 'Eduardo',
isAdmin: true
}),
actions: {
increment() {
this.counter++;
},
randomizeCounter() {
this.counter = Math.round(100 * Math.random());
}
}
});

View File

@ -0,0 +1,23 @@
export default {
namespaced: true,
state: () => ({
count: 0
}),
mutations: {
increment(state) {
state.count++;
}
},
getters: {
doubleCount(state) {
return state.count * 2;
}
},
actions: {
asyncIncrement({ commit }) {
setTimeout(() => {
commit('increment');
}, 2000);
}
}
};

View File

@ -0,0 +1,23 @@
export default {
namespaced: true,
state: () => ({
count: 0
}),
mutations: {
increment(state) {
state.count++;
}
},
getters: {
doubleCount(state) {
return state.count * 2;
}
},
actions: {
asyncIncrement({ commit }) {
setTimeout(() => {
commit('increment');
}, 2000);
}
}
};

View File

@ -0,0 +1,3 @@
import { createLogger } from 'vuex';
export default createLogger();

View File

@ -0,0 +1,54 @@
export default {
namespaced: true,
state: () => ({
name: 'aring',
age: 20,
count: 0
}),
mutations: {
increment(state) {
state.count++;
}
},
getters: {
doubleCount(state) {
return state.count * 2;
}
},
actions: {
asyncIncrement({ commit }) {
setTimeout(() => {
commit('increment');
}, 2000);
},
login() {
return new Promise((reslove) => {
setTimeout(() => {
console.log('login');
reslove('OK');
}, 1000);
});
}
},
modules: {
address: {
state: () => ({
province: '广东省',
city: '深圳市',
zone: '南山区'
}),
getters: {
address(state) {
return state.province + state.city + state.zone;
}
}
},
posts: {
namespaced: true,
state: () => ({}),
mutations: {
doSomething() {}
}
}
}
};

View File

@ -0,0 +1,3 @@
export default function sum(a, b) {
return a + b;
}

View File

@ -0,0 +1,38 @@
{
"compilerOptions": {
"outDir": "build/dist",
"module": "esnext",
"target": "esnext",
"lib": ["esnext", "dom"],
"sourceMap": true,
"baseUrl": ".",
"jsx": "preserve",
"allowSyntheticDefaultImports": true,
"moduleResolution": "node",
"forceConsistentCasingInFileNames": true,
"noImplicitReturns": true,
"suppressImplicitAnyIndexErrors": true,
"noUnusedLocals": true,
"allowJs": true,
"skipLibCheck": true,
"experimentalDecorators": true,
"strict": true,
"paths": {
"@/*": ["./src/*"],
"@@/*": ["./src/.fes/*"]
}
},
"include": [
"src/**/*",
"tests/**/*",
"test/**/*",
"__test__/**/*",
"typings/**/*",
"config/**/*",
".eslintrc.js",
".stylelintrc.js",
".prettierrc.js",
"src/.fes/configType.d.ts"
],
"exclude": ["node_modules", "build", "dist", "scripts", "src/.fes/*", "webpack", "jest"]
}

View File

@ -2,13 +2,10 @@ module.exports = {
extends: ['@webank/eslint-config-webank/vue.js'], extends: ['@webank/eslint-config-webank/vue.js'],
overrides: [ overrides: [
{ {
files: [ files: ['**/__tests__/*.{j,t}s?(x)', '**/tests/unit/**/*.spec.{j,t}s?(x)'],
'**/__tests__/*.{j,t}s?(x)', },
'**/tests/unit/**/*.spec.{j,t}s?(x)'
]
}
], ],
env: { env: {
jest: true jest: true,
} },
}; };

View File

@ -1,8 +1,6 @@
import { access as accessApi, pinia } from '@fesjs/fes'; import { access as accessApi, pinia } from '@fesjs/fes';
import PageLoading from '@/components/PageLoading'; import PageLoading from '@/components/PageLoading.vue';
import UserCenter from '@/components/UserCenter'; import UserCenter from '@/components/UserCenter.vue';
import { useStore } from '@/store/main'; import { useStore } from '@/store/main';
export const beforeRender = { export const beforeRender = {
@ -13,17 +11,17 @@ export const beforeRender = {
setTimeout(() => { setTimeout(() => {
const store = useStore(pinia); const store = useStore(pinia);
store.$patch({ store.$patch({
userName: '李雷' userName: '李雷',
}); });
setRole('admin'); setRole('admin');
resolve({ resolve({
userName: '李雷' userName: '李雷',
}); });
}, 1000); }, 1000);
}); });
} },
}; };
export const layout = { export const layout = {
customHeader: <UserCenter /> customHeader: <UserCenter />,
}; };

View File

@ -10,12 +10,11 @@ import { FButton } from '@fesjs/fes-design';
export default { export default {
components: { components: {
FButton FButton,
}, },
setup() { setup() {
return { return {};
}; },
}
}; };
</script> </script>

View File

@ -10607,6 +10607,11 @@ vite-plugin-html@^3.2.0:
node-html-parser "^5.3.3" node-html-parser "^5.3.3"
pathe "^0.2.0" pathe "^0.2.0"
vite-plugin-monaco-editor@^1.0.10:
version "1.0.10"
resolved "https://registry.npmmirror.com/vite-plugin-monaco-editor/-/vite-plugin-monaco-editor-1.0.10.tgz#cd370f68d4121bced6f902c6284649cc8eca4170"
integrity sha512-7yTAFIE0SefjCmfnjrvXOl53wkxeSASc/ZIcB5tZeEK3vAmHhveV8y3f90Vp8b+PYdbUipjqf91mbFbSENkpcw==
vite-plugin-windicss@^1.8.3: vite-plugin-windicss@^1.8.3:
version "1.8.3" version "1.8.3"
resolved "https://registry.npmmirror.com/vite-plugin-windicss/-/vite-plugin-windicss-1.8.3.tgz#d5fe923ad60f5d80f153a4fae5f837d56caa25cb" resolved "https://registry.npmmirror.com/vite-plugin-windicss/-/vite-plugin-windicss-1.8.3.tgz#d5fe923ad60f5d80f153a4fae5f837d56caa25cb"