Browse Source

init

master
1549469775 3 years ago
commit
25a7d913b9
  1. 7
      .gitignore
  2. 5
      .idea/.gitignore
  3. 49
      .idea/codeStyles/Project.xml
  4. 5
      .idea/codeStyles/codeStyleConfig.xml
  5. 12
      .idea/god-run.iml
  6. 8
      .idea/modules.xml
  7. 4
      .idea/watcherTasks.xml
  8. 11
      .prettierrc
  9. 53
      README.md
  10. 11
      components.d.ts
  11. 3
      docs/issues.md
  12. 24
      docs/思路.md
  13. 13
      index.html
  14. 41
      mock/login.ts
  15. 4049
      package-lock.json
  16. 33
      package.json
  17. 2189
      pnpm-lock.yaml
  18. BIN
      public/favicon.ico
  19. BIN
      public/static/icons/icon1.png
  20. BIN
      public/static/icons/icon2.png
  21. BIN
      public/static/icons/icon3.png
  22. BIN
      public/static/icons/sprites.png
  23. BIN
      public/static/login/bg.png
  24. BIN
      public/static/login/bg@2x.png
  25. BIN
      public/static/login/yun.png
  26. BIN
      public/static/login/yun@2x.png
  27. BIN
      public/static/logo.png
  28. BIN
      public/static/logo@2x.png
  29. 18
      src/App.vue
  30. 21
      src/api/_request/error.ts
  31. 55
      src/api/_request/index.ts
  32. 32
      src/api/_request/interceptors.ts
  33. 11
      src/api/contact/CompanyStruct.ts
  34. 9
      src/api/index.ts
  35. 17
      src/assets/script/util.ts
  36. 49
      src/assets/style/anim.less
  37. 167
      src/assets/style/common.less
  38. 27
      src/assets/style/css/anim.css
  39. 52
      src/assets/style/css/reset.css
  40. 2
      src/assets/style/global.less
  41. 39
      src/assets/style/less/media/exact.less
  42. 3
      src/assets/style/less/media/media.less
  43. 39
      src/assets/style/less/media/mobile.less
  44. 39
      src/assets/style/less/media/pc.less
  45. 48
      src/assets/style/less/row/row.less
  46. 58
      src/assets/style/less/util.less
  47. 5
      src/assets/style/less/var.less
  48. 30
      src/assets/style/less/visible/visible.less
  49. 82
      src/components/Toast/Toast.vue
  50. 35
      src/components/Toast/index.ts
  51. 16
      src/components/index.ts
  52. 1
      src/config/index.ts
  53. 20
      src/directive/index.ts
  54. 4
      src/enum/index.ts
  55. 7
      src/enum/page.ts
  56. 20
      src/env.d.ts
  57. 10
      src/hook/global/useRootSetting.ts
  58. 26
      src/hook/useDefineRoute.ts
  59. 14
      src/hook/useRouteChange.ts
  60. 26
      src/main.ts
  61. 9
      src/pages/Home/Home.vue
  62. 20
      src/pages/HomeRoute.ts
  63. 13
      src/pagesSys/Exception/Exception.vue
  64. 42
      src/pagesSys/Layout/Layout.vue
  65. 31
      src/pagesSys/Layout/SiteContent/SiteContent.vue
  66. 10
      src/pagesSys/Layout/SiteHeader/SiteHeader.vue
  67. 64
      src/pagesSys/Layout/SiteHeader/style.less
  68. 7
      src/pagesSys/Layout/SliderMenu/SliderMenu.vue
  69. 11
      src/pagesSys/Login/Login.vue
  70. 88
      src/pagesSys/Login/style.less
  71. 25
      src/pagesSys/Redirect/Redirect.vue
  72. 44
      src/pagesUI/editor/editor.vue
  73. 20
      src/pagesUI/editor/register.ts
  74. 25
      src/pagesUI/editor/work.ts
  75. 88
      src/pagesUI/verifyCodeBtn/verifyCodeBtn.vue
  76. 5
      src/plugins/index.md
  77. 13
      src/plugins/index.ts
  78. 95
      src/router/basic/index.ts
  79. 0
      src/router/basic/route/.gitkeep
  80. 12
      src/router/constant.ts
  81. 30
      src/router/guard/index.ts
  82. 20
      src/router/index.ts
  83. 8
      src/store/index.ts
  84. 13
      src/store/module/user.ts
  85. 23
      tsconfig.json
  86. 0
      types/index.d.ts
  87. 37
      vite.config.ts
  88. 3
      windi.config.ts

7
.gitignore

@ -0,0 +1,7 @@
node_modules
.DS_Store
dist
dist-ssr
*.local
src/pages/Ezopen
src/router/route/ezopen.ts

5
.idea/.gitignore

@ -0,0 +1,5 @@
# Default ignored files
/shelf/
/workspace.xml
# Editor-based HTTP Client requests
/httpRequests/

49
.idea/codeStyles/Project.xml

@ -0,0 +1,49 @@
<component name="ProjectCodeStyleConfiguration">
<code_scheme name="Project" version="173">
<HTMLCodeStyleSettings>
<option name="HTML_SPACE_INSIDE_EMPTY_TAG" value="true" />
<option name="HTML_ENFORCE_QUOTES" value="true" />
</HTMLCodeStyleSettings>
<JSCodeStyleSettings version="0">
<option name="USE_SEMICOLON_AFTER_STATEMENT" value="false" />
<option name="FORCE_SEMICOLON_STYLE" value="true" />
<option name="SPACE_BEFORE_FUNCTION_LEFT_PARENTH" value="false" />
<option name="FORCE_QUOTE_STYlE" value="true" />
<option name="ENFORCE_TRAILING_COMMA" value="Remove" />
<option name="SPACES_WITHIN_OBJECT_LITERAL_BRACES" value="true" />
<option name="SPACES_WITHIN_IMPORTS" value="true" />
</JSCodeStyleSettings>
<TypeScriptCodeStyleSettings version="0">
<option name="USE_SEMICOLON_AFTER_STATEMENT" value="false" />
<option name="FORCE_SEMICOLON_STYLE" value="true" />
<option name="SPACE_BEFORE_FUNCTION_LEFT_PARENTH" value="false" />
<option name="FORCE_QUOTE_STYlE" value="true" />
<option name="ENFORCE_TRAILING_COMMA" value="Remove" />
<option name="SPACES_WITHIN_OBJECT_LITERAL_BRACES" value="true" />
<option name="SPACES_WITHIN_IMPORTS" value="true" />
</TypeScriptCodeStyleSettings>
<VueCodeStyleSettings>
<option name="INTERPOLATION_NEW_LINE_AFTER_START_DELIMITER" value="false" />
<option name="INTERPOLATION_NEW_LINE_BEFORE_END_DELIMITER" value="false" />
</VueCodeStyleSettings>
<codeStyleSettings language="HTML">
<option name="SOFT_MARGINS" value="140" />
<indentOptions>
<option name="CONTINUATION_INDENT_SIZE" value="4" />
</indentOptions>
</codeStyleSettings>
<codeStyleSettings language="JavaScript">
<option name="SOFT_MARGINS" value="140" />
</codeStyleSettings>
<codeStyleSettings language="TypeScript">
<option name="SOFT_MARGINS" value="140" />
</codeStyleSettings>
<codeStyleSettings language="Vue">
<option name="SOFT_MARGINS" value="140" />
<indentOptions>
<option name="INDENT_SIZE" value="4" />
<option name="TAB_SIZE" value="4" />
</indentOptions>
</codeStyleSettings>
</code_scheme>
</component>

5
.idea/codeStyles/codeStyleConfig.xml

@ -0,0 +1,5 @@
<component name="ProjectCodeStyleConfiguration">
<state>
<option name="USE_PER_PROJECT_SETTINGS" value="true" />
</state>
</component>

12
.idea/god-run.iml

@ -0,0 +1,12 @@
<?xml version="1.0" encoding="UTF-8"?>
<module type="WEB_MODULE" version="4">
<component name="NewModuleRootManager">
<content url="file://$MODULE_DIR$">
<excludeFolder url="file://$MODULE_DIR$/temp" />
<excludeFolder url="file://$MODULE_DIR$/.tmp" />
<excludeFolder url="file://$MODULE_DIR$/tmp" />
</content>
<orderEntry type="inheritedJdk" />
<orderEntry type="sourceFolder" forTests="false" />
</component>
</module>

8
.idea/modules.xml

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="ProjectModuleManager">
<modules>
<module fileurl="file://$PROJECT_DIR$/.idea/god-run.iml" filepath="$PROJECT_DIR$/.idea/god-run.iml" />
</modules>
</component>
</project>

4
.idea/watcherTasks.xml

@ -0,0 +1,4 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="ProjectTasksOptions" suppressed-tasks="Less" />
</project>

11
.prettierrc

@ -0,0 +1,11 @@
{
"tabWidth": 4,
"useTabs": false,
"semi": false,
"singleQuote": false,
"TrailingCooma": "all",
"bracketSpacing": true,
"jsxBracketSameLine": false,
"arrowParens": "avoid",
"printWidth": 140
}

53
README.md

@ -0,0 +1,53 @@
# Vue 3 + Typescript + Vite
This template should help get you started developing with Vue 3 and Typescript in Vite. The template uses Vue 3 `<script setup>` SFCs, check out the [script setup docs](https://v3.vuejs.org/api/sfc-script-setup.html#sfc-script-setup) to learn more.
## Recommended IDE Setup
- [VSCode](https://code.visualstudio.com/) + [Volar](https://marketplace.visualstudio.com/items?itemName=johnsoncodehk.volar)
## Type Support For `.vue` Imports in TS
Since TypeScript cannot handle type information for `.vue` imports, they are shimmed to be a generic Vue component type by default. In most cases this is fine if you don't really care about component prop types outside of templates. However, if you wish to get actual prop types in `.vue` imports (for example to get props validation when using manual `h(...)` calls), you can enable Volar's `.vue` type support plugin by running `Volar: Switch TS Plugin on/off` from VSCode command palette.
https://github.com/anncwb/vue-vben-admin/blob/main/src/utils/cache/persistent.ts
https://v3.vuejs.org/api/sfc-script-setup.html#defineprops-and-defineemits
// Check out https://v3.vuejs.org/api/sfc-script-setup.html#sfc-script-setup
https://github.com/iiiiiii1/douban-imdb-api
https://www.jianshu.com/p/3f224c33dd97
http://doc.bmob.cn/data/wechat_app_new/#websocket
https://github.com/bmob/Bmob-wechatapp-xinyou
https://www.bmob.cn
https://www.bookstack.cn/read/BmobDocs/mds-other-common_problem-data_javascript.md
https://bmob.github.io/hydrogen-js-sdk/#/?id=%E7%99%BB%E9%99%86
https://v3.cn.vuejs.org/api/built-in-components.html#teleport
https://v3.cn.vuejs.org/api/global-api.html#defineasynccomponent
https://2x.antdv.com/docs/vue/introduce-cn
https://www.attojs.com/api/#manual
## 目录结构说明
```
src
├── api (网络请求模块)
├── App.vue
├── assets (资源模块,包含公共文件)
├── components (共用模块,与业务无关)
├── env.d.ts
├── hook (存放通用hook)
├── main.ts
├── pages (页面)
├── pagesUI (通用页面组件)
├── plugins (其他插件)
├── router (路由管理)
└── store (全局状态管理)
```
## 注意事项
* 页面节点必须只有一个根节点,不然会发现动画出现问题

11
components.d.ts

@ -0,0 +1,11 @@
// generated by unplugin-vue-components
// We suggest you to commit this file into source control
// Read more: https://github.com/vuejs/vue-next/pull/3399
declare module 'vue' {
export interface GlobalComponents {
Toast: typeof import('./src/components/Toast/Toast.vue')['default']
}
}
export { }

3
docs/issues.md

@ -0,0 +1,3 @@
### (close) vue3 + typescript + elementPlus生产包打包报错
https://blog.51cto.com/youthfighter/2765511

24
docs/思路.md

@ -0,0 +1,24 @@
# 布局思路
根据UI图的设计,总体可以分为两层或者三层布局
* 两层布局
* 登录页面,工作页面,此为一层布局
* 工作页面分为顶部标题栏和下面的工作区域
* 缺点
* 要控制工作区域的侧边菜单显隐与右侧内容的动画,导致会出现闪动的画面
* 两层布局+布局切换
* 登录页面,工作页面,此为一层布局
* 工作页面可以分为两部分:
* 顶部标题栏和下面的工作区域
* 顶部标题栏+侧边菜单+下面的工作区域
* 缺点
* keep-alive缓存不全
* 三层布局
* 登录页面,工作页面,此为一层布局
* 工作台页面以及功能页面,此为第二层布局
* 功能页面分为菜单+右侧内容,此为第三层布局
* 缺点:
* 需要两层keep-alive才能覆盖整个的页面,同时会导致多了一层路由
因为不太想包三层布局,因此采用上面的两层布局。

13
index.html

@ -0,0 +1,13 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<link rel="icon" href="/favicon.ico" />
<meta name="viewport" content="width=device-width, initial-scale=1,maximum-scale=1,user-scalable=no">
<title>Vite App</title>
</head>
<body>
<div id="app"></div>
<script type="module" src="/src/main.ts"></script>
</body>
</html>

41
mock/login.ts

@ -0,0 +1,41 @@
import { MockMethod } from 'vite-plugin-mock'
export default [
{
url: '/api/login',
method: 'get',
response: ({ query }) => {
return {
code: 0,
data: {
token: 'sadw@fjdsfsdkfsd3g6ujhj8i54wdvu7un7sadsaa',
},
}
},
},
{
url: '/api/sendcode',
method: 'post',
timeout: 1500,
response: {
code: 0,
data: null,
message: 'success',
},
},
{
url: '/api/text',
method: 'post',
rawResponse: async (req, res) => {
let reqbody = ''
await new Promise((resolve) => {
req.on('data', (chunk) => {
reqbody += chunk
})
req.on('end', () => resolve(undefined))
})
res.setHeader('Content-Type', 'text/plain')
res.statusCode = 200
res.end(`hello, ${reqbody}`)
},
},
] as MockMethod[]

4049
package-lock.json

File diff suppressed because it is too large

33
package.json

@ -0,0 +1,33 @@
{
"name": "vite-project",
"version": "0.0.0",
"scripts": {
"dev": "vite --host",
"build": "vue-tsc --noEmit && vite build",
"serve": "vite preview"
},
"dependencies": {
"@vueuse/core": "^6.8.0",
"axios": "^0.24.0",
"lodash": "^4.17.21",
"mockjs": "^1.1.0",
"monaco-editor": "^0.30.1",
"pinia": "^2.0.2",
"unplugin-vue-components": "^0.17.0",
"vue": "^3.2.16",
"vue-request": "^1.2.3",
"vue-router": "^4.0.12"
},
"devDependencies": {
"@types/crypto-js": "^4.0.2",
"@types/node": "^16.11.6",
"@vitejs/plugin-vue": "^1.9.3",
"less": "^4.1.2",
"typescript": "^4.4.3",
"vite": "^2.6.14",
"vite-plugin-mock": "^2.9.6",
"vite-plugin-windicss": "^1.5.1",
"vue-tsc": "^0.3.0",
"windicss": "^3.2.1"
}
}

2189
pnpm-lock.yaml

File diff suppressed because it is too large

BIN
public/favicon.ico

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.2 KiB

BIN
public/static/icons/icon1.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 750 B

BIN
public/static/icons/icon2.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 881 B

BIN
public/static/icons/icon3.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 732 B

BIN
public/static/icons/sprites.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 19 KiB

BIN
public/static/login/bg.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 122 KiB

BIN
public/static/login/bg@2x.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 258 KiB

BIN
public/static/login/yun.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 65 KiB

BIN
public/static/login/yun@2x.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 110 KiB

BIN
public/static/logo.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.2 KiB

BIN
public/static/logo@2x.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 13 KiB

18
src/App.vue

@ -0,0 +1,18 @@
<script setup lang="ts">
import { useTitle } from '@vueuse/core'
import { watchEffect } from 'vue'
import { useRoute } from 'vue-router'
const route = useRoute()
watchEffect(() => {
let title = route.meta.title as string
useTitle(title || '爱能森平台')
})
</script>
<template>
<router-view></router-view>
</template>
<style lang="less">
@import '@/assets/style/common.less';
</style>

21
src/api/_request/error.ts

@ -0,0 +1,21 @@
import { message } from 'ant-design-vue'
import { userStore } from "@/store/module/user"
export default function error(err: any, type: number) {
switch (type) {
case 0:
const store = userStore()
message.error(err?.data?.message?err?.data?.message:"未知错误")
if(err?.data&&err?.data.code === -1){
store.logout(false)
}
// 响应正确,服务器返回错误
break
case 1:
// 服务器错误
break
case 2:
// 请求错误
break
}
}

55
src/api/_request/index.ts

@ -0,0 +1,55 @@
import { IHttpMethod } from '@/enum'
import axios from 'axios'
import checkError from './error'
import { BASE_URL } from '@/config'
import createInterceptors from './interceptors'
const urlRegexp = new RegExp(/^((https|http|ftp|rtsp|mms)?:\/\/)[^\s]+/i)
const axiosInstance = axios.create({
timeout: 5000,
})
createInterceptors(axiosInstance)
function http(
url: string,
method: IHttpMethod = IHttpMethod.GET,
data: any = {}
): Promise<any> {
var __data: { params?: any; data?: any } = {}
if (method === IHttpMethod.GET) {
__data.params = data
}
if (method === IHttpMethod.POST) {
__data.data = data
}
let realurl = urlRegexp.test(url) ? url : BASE_URL + url
return new Promise((resolve, reject) => {
axiosInstance({
url: realurl,
method,
headers: {},
responseType: 'json',
...__data,
})
.then((response) => {
if (response.status == 200) {
if (response.data!=undefined && response.data.code === 0) {
resolve(response.data)
} else {
checkError(response, 0)
reject(response)
}
} else {
checkError(response, 1)
reject(response)
}
})
.catch((error) => {
checkError(error, 2)
reject(error)
})
})
}
export { http }

32
src/api/_request/interceptors.ts

@ -0,0 +1,32 @@
import type { AxiosInstance, AxiosRequestConfig } from 'axios'
import { userStore } from '@/store/module/user'
export default function createInterceptors(axios: AxiosInstance) {
axios.interceptors.request.use(
function (config: AxiosRequestConfig) {
const store = userStore()
if (config && config.headers&&store.token){
config.headers['token'] = store.token
}
if (config && config.headers && !config!.headers['Content-Type']) {
config.headers['Accept'] = 'application/json';
config.headers['Content-Type'] = 'application/x-www-form-urlencoded';
}
// 在发送请求之前做些什么
return config
},
function (error) {
// 对请求错误做些什么
return Promise.reject(error)
}
)
axios.interceptors.response.use(
function (response) {
return response
},
function (error) {
return Promise.reject(error)
}
)
}

11
src/api/contact/CompanyStruct.ts

@ -0,0 +1,11 @@
import { http } from "@/api/_request"
import { IHttpMethod } from "@/enum"
import { readonly } from "vue"
const api = {
getStruct: (data?: any) => http("/enesoon/organization/v1.0/orgStructure", IHttpMethod.GET, data),
getDepartment: (data?: any) => http("/enesoon/organization/v1.0/orgDepartment", IHttpMethod.GET, data),
getDepartmentEmp: (data?: any) => http("/enesoon/organization/v1.0/departmentEmp", IHttpMethod.GET, data),
}
export default api

9
src/api/index.ts

@ -0,0 +1,9 @@
import { IHttpMethod } from '@/enum'
import { http } from './_request'
const api = {
login: (data?: any) => http('/enesoon/login/v1.0/employees', IHttpMethod.GET, data),
getUserProfile: (data?: any) => http('/enesoon/user/v1.0/employees', IHttpMethod.GET, data)
}
export default api

17
src/assets/script/util.ts

@ -0,0 +1,17 @@
export function isPromise(obj: any) {
return !!obj && (typeof obj === 'object' || typeof obj === 'function') && typeof obj.then === 'function';
}
export async function execute(fn: () => any) {
let result;
if(isPromise(fn)){
result = await fn()
}else{
result = fn()
}
return result
}
export default {}

49
src/assets/style/anim.less

@ -0,0 +1,49 @@
.fade-enter-active,
.fade-leave-active {
transition: opacity .25s ease;
}
.fade-enter-from,
.fade-leave-to {
opacity: 0;
}
.slide-y-transition {
.transition-default();
&-enter-from,
&-leave-to {
opacity: 0%;
transform: translateY(-15px);
}
}
.slide-y-reverse-transition {
.transition-default();
&-enter-from,
&-leave-to {
opacity: 0%;
transform: translateY(15px);
}
}
.slide-x-transition {
.transition-default();
&-enter-from,
&-leave-to {
opacity: 0%;
transform: translateX(-15px);
}
}
.slide-x-reverse-transition {
.transition-default();
&-enter-from,
&-leave-to {
opacity: 0%;
transform: translateX(15px);
}
}

167
src/assets/style/common.less

@ -0,0 +1,167 @@
@import (inline) './css/reset.css';
@import (inline) './css/anim.css';
@import "./anim.less";
@import "./less/row/row.less";
@import "./less/visible/visible.less";
html,
body {
// background-color: #43b244;
// color: #fffef8;
margin: 0;
padding: 0;
}
html,
body{
width: 100%;
height: 100%;
background-color: #f9f9f9 !important;
overflow: auto !important;
overflow-x: auto !important;
}
#app {
width: 100%;
height: 100%;
}
body {
font: 12px/1.5 Tahoma, Helvetica, Arial, '宋体', sans-serif;
}
*,
::before,
::after {
box-sizing: border-box;
}
.container {
margin: 0 auto;
max-width: 1200px;
}
.clearfix {
.clearfix();
}
.hairline--all() {
.hairline({
border-width: 1px;
});
}
.hairline {
.hairline--all();
}
.ellipsis {
.ellipsis();
}
.over-ellipsis {
display: -webkit-box;
-webkit-box-orient: vertical;
-webkit-line-clamp: 2;
overflow: hidden;
text-overflow: ellipsis;
word-break: break-all;
}
.a-flexbox{
display: flex;
&.a-flexbox--fixed{
flex-shrink: 0;
flex-grow: 0;
}
&.a-flexbox--inline{
display: inline-flex;
}
&.a-flexbox__column{
flex-direction: column;
}
&.a-flexbox__row{
flex-direction: row;
}
&.a-flexbox__wrap{
flex-wrap: wrap;
}
&.a-flexbox__center{
align-items: center;
justify-content: center;
}
&.a-flexbox__center--x{
justify-content: center;
}
&.a-flexbox--around{
justify-content: space-around;
}
&.a-flexbox--between{
justify-content: space-between;
}
&.a-flexbox__start--x{
justify-content: flex-start;
}
&.a-flexbox__end--x{
justify-content: flex-end;
}
&.a-flexbox__center--y{
align-items: center;
}
&.a-flexbox__start--y{
align-items: flex-start;
}
&.a-flexbox__end--y{
align-items: flex-end;
}
.a-flexbox__noshrink{
flex-shrink: 0;
}
.a-flexbox__full{
flex: 1;
width: 0;
}
.a-flexbox__full--h{
flex: 1;
height: 0;
}
}
.transition-default() {
&-enter-active,
&-leave-active {
transition: 0.3s cubic-bezier(0.25, 0.8, 0.5, 1) !important;
}
&-move {
transition: transform 0.4s;
}
}
// html {
// font-size : 20px;
// }
// @media only screen and (min-width: 401px){
// html {
// font-size: 25px !important;
// }
// }
// @media only screen and (min-width: 428px){
// html {
// font-size: 26.75px !important;
// }
// }
// @media only screen and (min-width: 481px){
// html {
// font-size: 30px !important;
// }
// }
// @media only screen and (min-width: 569px){
// html {
// font-size: 35px !important;
// }
// }
// @media only screen and (min-width: 641px){
// html {
// font-size: 40px !important;
// }
// }

27
src/assets/style/css/anim.css

@ -0,0 +1,27 @@
.-slidein-y{
animation: slide-in .5s ease;
}
.-opacityin{
animation: slide-in .5s ease;
}
@keyframes slide-in {
0%{
transform: translateY(-120%);
opacity: 0;
}
100%{
transform: translateY(0);
opacity: 1;
}
}
@keyframes opacity-in {
0%{
opacity: 0;
}
100%{
opacity: 1;
}
}

52
src/assets/style/css/reset.css

@ -0,0 +1,52 @@
/* http://meyerweb.com/eric/tools/css/reset/
v2.0 | 20110126
License: none (public domain)
*/
html, body, div, span, applet, object, iframe,
h1, h2, h3, h4, h5, h6, p, blockquote, pre,
a, abbr, acronym, address, big, cite, code,
del, dfn, em, img, ins, kbd, q, s, samp,
small, strike, strong, sub, sup, tt, var,
b, u, i, center,
dl, dt, dd, ol, ul, li,
fieldset, form, label, legend,
table, caption, tbody, tfoot, thead, tr, th, td,
article, aside, canvas, details, embed,
figure, figcaption, footer, header, hgroup,
menu, nav, output, ruby, section, summary,
time, mark, audio, video {
margin: 0;
padding: 0;
border: 0;
font-size: 100%;
font: inherit;
vertical-align: baseline;
}
/* HTML5 display-role reset for older browsers */
article, aside, details, figcaption, figure,
footer, header, hgroup, menu, nav, section {
display: block;
}
body {
line-height: 1;
}
ol, ul {
list-style: none;
}
blockquote, q {
quotes: none;
}
blockquote:before, blockquote:after,
q:before, q:after {
content: '';
content: none;
}
table {
border-collapse: collapse;
border-spacing: 0;
}
a{
text-decoration: none;
color: inherit;
}

2
src/assets/style/global.less

@ -0,0 +1,2 @@
@import './less/media/media.less';
@import './less/util.less';

39
src/assets/style/less/media/exact.less

@ -0,0 +1,39 @@
/**
* 响应式 媒体查询类
**/
@import (reference) "../var.less";
.choose3(@type, @style) when (@type=xs){
@media screen and (max-width: @sm - 0.1px){
@style();
}
}
.choose3(@type, @style) when (@type=sm){
@media screen and (min-width: @sm - 0.1px) and (max-width: @md - 0.1px){
@style();
}
}
.choose3(@type, @style) when (@type=md){
@media screen and (min-width: @md - 0.1px) and (max-width: @lg - 0.1px){
@style();
}
}
.choose3(@type, @style) when (@type=lg){
@media screen and (min-width: @lg - 0.1px) and (max-width: @xl - 0.1px){
@style();
}
}
.choose3(@type, @style) when (@type=xl){
@media screen and (min-width: @xl - 0.1px) {
@style();
}
}
.loop3(@i,@style,@list) when (@i<length(@list)){
.choose3(extract(@list,@i+1),@style);
.loop3(@i+1,@style,@list);
}
.media_e(@style,@rest...){
.loop3(0,@style,@rest);
}

3
src/assets/style/less/media/media.less

@ -0,0 +1,3 @@
@import (reference) "./exact.less";
@import (reference) "./mobile.less";
@import (reference) "./pc.less";

39
src/assets/style/less/media/mobile.less

@ -0,0 +1,39 @@
/**
* 响应式 媒体查询类
* 移动端优先
**/
@import (reference) "../var.less";
.choose2(@type, @style) when (@type=xs){
@media screen and (max-width: @sm - 0.1px){
@style();
}
}
.choose2(@type, @style) when (@type=sm){
@media screen and (max-width: @md - 0.1px){
@style();
}
}
.choose2(@type, @style) when (@type=md){
@media screen and (max-width: @lg - 0.1px) {
@style();
}
}
.choose2(@type, @style) when (@type=lg){
@media screen and (max-width: @xl - 0.1px) {
@style();
}
}
.choose2(@type, @style) when (@type=xl){
@media screen and (min-width: @xl - 0.1px) {
@style();
}
}
.loop2(@i,@style,@list) when (@i<length(@list)){
.choose2(extract(@list,@i+1),@style);
.loop2(@i+1,@style,@list);
}
.media_m(@style,@rest...){
.loop2(0,@style,@rest);
}

39
src/assets/style/less/media/pc.less

@ -0,0 +1,39 @@
/**
* 响应式 媒体查询类
**/
@import (reference) "../var.less";
.choose1(@type, @style) when (@type=xs){
@media screen and (max-width: @sm){
@style();
}
}
.choose1(@type, @style) when (@type=sm){
@media screen and (min-width: @sm){
@style();
}
}
.choose1(@type, @style) when (@type=md){
@media screen and (min-width: @md){
@style();
}
}
.choose1(@type, @style) when (@type=lg){
@media screen and (min-width: @lg) {
@style();
}
}
.choose1(@type, @style) when (@type=xl){
@media screen and (min-width: @xl) {
@style();
}
}
.loop1(@i,@style,@list) when (@i<length(@list)){
.choose1(extract(@list,@i+1),@style);
.loop1(@i+1,@style,@list);
}
.media(@style,@rest...){
.loop1(0,@style,@rest);
}

48
src/assets/style/less/row/row.less

@ -0,0 +1,48 @@
/**
* 响应式布局类
**/
@totalCell: 24;
.col-width(@col: 8, @total: @totalCell) {
width: (@col / @total) * 100%;
}
.col-offset(@col: 8, @total: @totalCell) {
margin-left: (@col / @total) * 100%;
}
.col(@i, @mv) when (@i > 0) {
.col((@i - 1), @mv); // 递归调用自身
.col-@{mv}-@{i} {
float: left;
box-sizing: border-box;
.col-width(@i);
}
}
.colr(@i) when (@i > 0) {
.colr((@i - 1)); // 递归调用自身
.col-@{i} {
float: left;
box-sizing: border-box;
.col-width(@i);
}
.col-offset-@{i} {
.col-offset(@i);
}
}
.row {
.clearfix();
.colr(@totalCell);
.media({
.col(@totalCell, lg);
}, lg);
.media({
.col(@totalCell, md);
}, md);
.media({
.col(@totalCell, sm);
}, sm);
.media({
.col(@totalCell, xs);
}, xs);
}

58
src/assets/style/less/util.less

@ -0,0 +1,58 @@
/**
* 工具类
**/
/**
* START============清除浮动=================START
**/
.clearfix() {
&:after {
clear: both;
content: ".";
display: block;
height: 0;
line-height: 0;
overflow: hidden;
}
// *height: 1%;
}
/**
* OVER=============清除浮动==================OVER
**/
/**
* START============移动端头发丝=================START
**/
.hairline(@content) {
&::after {
content: " ";
box-sizing: border-box;
pointer-events: none;
border: 0 solid #ebedf0;
position: absolute;
top: -50%;
right: -50%;
bottom: -50%;
left: -50%;
-webkit-transform: scale(0.5);
transform: scale(0.5);
@content();
}
}
/**
* OVER=============移动端头发丝==================OVER
**/
/**
* START============文本溢出省略号=================START
**/
.ellipsis(@lines: 1) when (@lines = 1) {
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
/**
* OVER=============文本溢出省略号==================OVER
**/

5
src/assets/style/less/var.less

@ -0,0 +1,5 @@
@sm: 768px;
@md: 992px;
@lg: 1200px;
@xl: 1920px;

30
src/assets/style/less/visible/visible.less

@ -0,0 +1,30 @@
/**
* 响应式类,控制元素的显示隐藏
**/
@import (reference) "../var.less";
@xs-only: e('max-width: @{sm}');
@sm-and-up: e('min-width: @{sm}');
@sm-only: e('min-width: @{sm}) and (max-width: @{md}');
@sm-and-down: e('max-width: @{md}');
@md-and-up: e('min-width: @{md}');
@md-only: e('min-width: @{md}) and (max-width: @{lg}');
@md-and-down: e('max-width: @{lg}');
@lg-and-up: e('min-width: @{lg}');
@lg-only: e('min-width: @{lg}) and (max-width: @{xl}');
@lg-and-down: e('max-width: @{xl}');
@xl-only: e('min-width: @{xl}');
@hide-array: xs-only, sm-and-up, sm-only, sm-and-down, md-and-up, md-only, md-and-down, lg-and-up, lg-only, lg-and-down, xl-only;
.hide(@i: 1) when(@i <= length(@hide-array)) {
@name: extract(@hide-array, @i);
.hidden-@{name} {
@media screen and (@@name) {
display: none !important;
}
}
.hide((@i + 1));
}
.hide();

82
src/components/Toast/Toast.vue

@ -0,0 +1,82 @@
<template>
<div>
<transition
name="fade"
@before-leave="emit('close')"
@after-leave="emit('destory')"
>
<div v-if="isShow" class="toast component">
<div class="toast-text">{{ message }}</div>
</div>
</transition>
</div>
</template>
<script setup lang="ts">
import { onMounted, ref } from 'vue'
const isShow = ref(false)
const props = withDefaults(
defineProps<{
message: string
duration?: number
}>(),
{
message: '',
duration: 3000,
}
)
const emit = defineEmits<{
(e: 'destory'): void
(e: 'close'): void
}>()
onMounted(() => {
isShow.value = true
startTimer()
})
let timer = null
function startTimer() {
if (props.duration) {
timer = setTimeout(() => {
close()
}, props.duration)
}
}
function close() {
isShow.value = false
}
</script>
<style lang="less" scoped>
.toast.component {
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
background-color: rgba(70, 70, 70, 0.8);
border-radius: 5px;
word-break: break-all;
max-width: 90vw;
text-align: center;
.toast-text {
user-select: none;
color: red;
line-height: 1.5;
font-size: 16px;
padding: 4px 10px;
}
}
.fade-enter-active,
.fade-leave-active {
transition: opacity 0.5s ease;
}
.fade-enter-from,
.fade-leave-to {
opacity: 0;
}
</style>

35
src/components/Toast/index.ts

@ -0,0 +1,35 @@
import { createVNode, render, isVNode } from 'vue'
import ToastConstructor from './Toast.vue'
type IProp = {
message: string
duration?: number
}
export type IOps = IProp | string
function createToast(options: IProp) {
const container = document.createElement('div')
container.className = `container_toast`
const vm = createVNode(ToastConstructor, options, null)
render(vm, container)
// @ts-ignore
vm.props.onDestory = () => {
render(null, container)
}
document.body.appendChild(container.firstElementChild as Element)
}
function Toast(opts: IOps) {
if (typeof opts === 'string') {
opts = {
message: opts,
}
}
let options = opts
createToast(options)
}
export default Toast

16
src/components/index.ts

@ -0,0 +1,16 @@
import { App } from 'vue'
import Row from './Row/Row.vue'
import Column from './Column/Column.vue'
// https://vitejs.dev/guide/features.html#glob-import
export default {
install(app: App) {
const modules = import.meta.globEager('./{icons,base}/**/*.vue')
for (const path in modules) {
const mod = modules[path]
const module = mod.default || mod
// let filename = module.__file.slice(module.__file.lastIndexOf("\/")+1,module.__file.lastIndexOf(".vue"))
app.component(module.name, module)
}
},
}

1
src/config/index.ts

@ -0,0 +1 @@
export const BASE_URL = "https://enesoon-saas-back-test.cn:8381"

20
src/directive/index.ts

@ -0,0 +1,20 @@
import { App, ref } from 'vue'
const isAuth = ref(false)
setTimeout(()=>{
isAuth.value = true
}, 2500)
export default {
install(app: App) {
app.directive('permission', (el: HTMLDivElement, binding) => {
const { arg, value} = binding
console.log(isAuth)
if (!isAuth.value) {
// el.remove()
el.parentNode?.removeChild(el)
}else {
el.parentNode?.appendChild(el)
}
})
},
}

4
src/enum/index.ts

@ -0,0 +1,4 @@
export enum IHttpMethod{
GET="GET",
POST="POST"
}

7
src/enum/page.ts

@ -0,0 +1,7 @@
enum EPage{
HOME,
}
export default EPage

20
src/env.d.ts

@ -0,0 +1,20 @@
/// <reference types="vite/client" />
declare module '*.vue' {
import { DefineComponent } from 'vue'
// eslint-disable-next-line @typescript-eslint/no-explicit-any, @typescript-eslint/ban-types
const component: DefineComponent<{}, {}, any>
export default component
}
interface ImportMetaEnv {
readonly VITE_SENTRY_URL: string
readonly VITE_SENTRY_TOKEN: string
readonly VITE_SENTRY_ORG: string
readonly VITE_SENTRY_PROJECT: string
readonly VITE_SENTRY_RELEASE: string
}
interface ImportMeta {
readonly env: ImportMetaEnv
}

10
src/hook/global/useRootSetting.ts

@ -0,0 +1,10 @@
import { ref } from "vue";
export default function useRootSetting() {
const openCache = ref(true)
return {
openCache
}
}

26
src/hook/useDefineRoute.ts

@ -0,0 +1,26 @@
import { RouteRecordRaw } from "vue-router"
export function useDefineRoute(route: RouteRecordRaw) {
return route
}
export function useDefineRoutes(
routes: RouteRecordRaw[]
) {
return routes
}
type Module = { default: RouteRecordRaw } | { default: RouteRecordRaw[] };
export function useRouteConfig(paths: Record<string, any>) {
let routes:RouteRecordRaw[] = []
for (const path in paths) {
const mod = paths[path]
const module = mod.default || mod
if (Array.isArray(module)) {
routes = module.concat(module)
} else {
routes.push(module)
}
}
return routes
}

14
src/hook/useRouteChange.ts

@ -0,0 +1,14 @@
import { nextTick, watchEffect } from 'vue'
import { useRoute } from 'vue-router'
export default function useRouteChange(cb:()=>void, waitRender?: boolean) {
const route = useRoute()
watchEffect(async () => {
if (route.fullPath) {
if(waitRender){
await nextTick()
}
cb&&cb()
}
})
}

26
src/main.ts

@ -0,0 +1,26 @@
import { createApp } from "vue"
import Router from "@/router"
import Store from "@/store"
import Directive from "@/directive"
import Plugins from "@/plugins"
import Components from "@/components"
import App from "./App.vue"
// import 'virtual:windi.css'
import 'virtual:windi-base.css'
import 'virtual:windi-components.css'
import 'virtual:windi-utilities.css'
import { setGlobalOptions } from "vue-request"
setGlobalOptions({
manual: true, // true:必须手动执行run触发 false:自动执行
})
const app = createApp(App)
app.use(Store)
app.use(Router)
app.use(Directive)
app.use(Plugins)
app.use(Components)
app.mount("#app")

9
src/pages/Home/Home.vue

@ -0,0 +1,9 @@
<template>
<div class="font-bold text-red-500">
<MonacoEditor></MonacoEditor>
</div>
</template>
<script lang="ts" setup>
import MonacoEditor from '@/pagesUI/editor/editor.vue';
</script>

20
src/pages/HomeRoute.ts

@ -0,0 +1,20 @@
import { useDefineRoutes, useRouteConfig } from "@/hook/useDefineRoute"
import { LAYOUT } from "@/router/constant"
export default useDefineRoutes([
{
path: '',
redirect: '/home',
component: LAYOUT,
children: [
{
path: 'home',
name: "HOME",
component: ()=>import("@/pages/Home/Home.vue"),
meta: {
title: "HOME"
}
}
]
}
])

13
src/pagesSys/Exception/Exception.vue

@ -0,0 +1,13 @@
<template>
<div>404 expection</div>
</template>
<script lang="ts" setup>
import { useRoute } from 'vue-router'
import { watchEffect } from 'vue'
const route = useRoute()
watchEffect(() => {
// queryprops
console.log(route.query)
})
</script>

42
src/pagesSys/Layout/Layout.vue

@ -0,0 +1,42 @@
<script lang="ts" setup>
import SiteHeader from "./SiteHeader/SiteHeader.vue"
import SiteContent from "./SiteContent/SiteContent.vue"
import SliderMenu from "./SliderMenu/SliderMenu.vue"
</script>
<template>
<section class="app-layout">
<SiteHeader></SiteHeader>
<div class="app-container">
<SliderMenu></SliderMenu>
<div class="app-content">
<SiteContent></SiteContent>
</div>
</div>
</section>
</template>
<style lang="less" scoped>
.app-layout {
height: 100%;
display: flex;
flex-direction: column;
.app-container {
flex: 1;
height: 0;
display: flex;
.app-slider {
flex-shrink: 0;
overflow-x: hidden;
overflow-y: auto;
background-color: #f9f9f9;
}
.app-content {
flex: 1;
width: 0;
overflow-x: hidden;
overflow-y: auto;
}
}
}
</style>

31
src/pagesSys/Layout/SiteContent/SiteContent.vue

@ -0,0 +1,31 @@
<template>
<router-view v-slot="{ Component, route: fuck }">
<!-- 暂时不搞动画布局不相同不太好看 -->
<!-- mode="out-in" 会导致热更新之后切换页面无效 或者说是 keep-alive-->
<!-- <transition :name="getTransitionName(fuck)" mode="out-in" appear @after-leave="afterLeave"> -->
<keep-alive :include="cacheList" v-if="openCache">
<component :key="route.fullPath" :is="Component" />
</keep-alive>
<component v-else :key="route.fullPath" :is="Component" />
<!-- </transition> -->
</router-view>
</template>
<script lang="ts" setup>
import { useRoute } from "vue-router"
import { ref, unref, watchEffect } from "vue"
const route = useRoute()
const openCache = ref(true)
const cacheList = ref<any[]>([])
watchEffect(() => {
if (
route.meta.cache &&
route.name &&
!cacheList.value.includes(unref(route.name))
) {
cacheList.value.push(unref(route.name))
}
})
</script>

10
src/pagesSys/Layout/SiteHeader/SiteHeader.vue

@ -0,0 +1,10 @@
<template>
<div>header</div>
</template>
<script lang="ts" setup>
</script>
<style lang="less" scoped>
@import "./style.less";
</style>

64
src/pagesSys/Layout/SiteHeader/style.less

@ -0,0 +1,64 @@
.site-header {
.site-header__wrapper {
height: 60px;
line-height: 60px;
padding: 0 40px;
background: #fefefe;
box-shadow: 0px 2px 4px 0px rgba(0, 0, 0, 0.06);
position: relative;
z-index: 10;
.site_header__logo {
float: left;
position: relative;
height: 100%;
padding-right: 14px;
img {
vertical-align: middle;
}
&::after {
content: '';
height: 20px;
width: 1px;
position: absolute;
right: 0;
top: 50%;
transform: translateY(-50%);
background-color: #d7dde4;
}
}
.site_header__menu {
margin-left: 22px;
height: 100%;
&.site_header__left {
float: left;
}
&.site_header__right {
float: right;
}
.sh__menu__item {
float: left;
height: 100%;
cursor: pointer;
padding: 0 10px;
position: relative;
font-size: 12px;
font-weight: 400;
color: #292929;
display: flex;
align-items: center;
justify-content: center;
a {
color: #292929;
}
&.active .shmi__text {
color: #24a439;
}
&:hover .shmi__text {
color: #24a439;
}
}
}
}
}

7
src/pagesSys/Layout/SliderMenu/SliderMenu.vue

@ -0,0 +1,7 @@
<template>
<div>slider-menu</div>
</template>
<script lang="ts" setup>
</script>

11
src/pagesSys/Login/Login.vue

@ -0,0 +1,11 @@
<template>
<div>login</div>
</template>
<script lang="ts" setup>
</script>
<style lang="less" scoped>
@import "./style.less";
</style>

88
src/pagesSys/Login/style.less

@ -0,0 +1,88 @@
.page {
min-height: 100%;
background: linear-gradient(164deg, #f0f4fb 0%, #dfe6f3 100%);
.header {
height: 60px;
line-height: 60px;
background: #fefefe;
.header__wrapper {
.media_m({ padding: 0 15px; },md);
height: 100%;
img {
vertical-align: middle;
}
}
}
.content {
margin-top: 93px;
.media_m({ padding: 0 15px; },md);
.left-panel {
margin-right: 17px;
.site-title {
font-size: 36px;
font-weight: 500;
color: #2e2e2f;
}
.site-desc {
margin-top: 5px;
font-size: 18px;
font-weight: 500;
color: #9e9e9e;
}
.site-picture {
margin-top: 70px;
position: relative;
.site-picture-bg {
position: absolute;
left: -229px;
top: -150px;
}
}
}
.login-panel {
background: #ffffff;
box-shadow: 0px 6px 4px 0px #e5eaef;
border-radius: 6px;
padding: 80px 48px 136px;
overflow: hidden;
transition: all 0.5s ease;
.login-text {
// margin-left: 8px;
font-size: 18px;
font-weight: 500;
color: #0d0d0e;
margin-bottom: 20px;
}
.button {
cursor: pointer;
height: 45px;
text-align: center;
font-size: 14px;
font-weight: 400;
color: #ffffff;
display: block;
width: 100%;
outline: 0;
border: 0;
background: linear-gradient(180deg, #32c14d 0%, #3abe81 100%);
}
.forget-pwd {
display: block;
text-align: right;
font-size: 14px;
font-weight: 400;
color: #3bad4e;
}
}
.copyright {
font-size: 11px;
font-weight: 500;
color: #6d6d6d;
padding-top: 89px;
.media_m({ padding-top: 25px; },md);
text-align: center;
position: relative;
}
}
}

25
src/pagesSys/Redirect/Redirect.vue

@ -0,0 +1,25 @@
<template>
<div></div>
</template>
<script lang="ts" setup>
import { unref } from 'vue';
import { useRouter } from 'vue-router';
const { currentRoute, replace } = useRouter();
const { params, query } = unref(currentRoute);
const { path, _redirect_type = 'path' } = params;
Reflect.deleteProperty(params, '_redirect_type');
Reflect.deleteProperty(params, 'path');
const _path = Array.isArray(path) ? path.join('/') : path;
if (_redirect_type === 'name') {
replace({
name: _path,
query,
params,
});
} else {
replace({
path: _path.startsWith('/') ? _path : '/' + _path,
query,
});
}
</script>

44
src/pagesUI/editor/editor.vue

@ -0,0 +1,44 @@
<template>
<div class="ui-monaco-editor" ref="monacoEL">
</div>
</template>
<script lang="ts" setup>
import 'monaco-editor/esm/vs/editor/editor.all.js';
import 'monaco-editor/esm/vs/editor/standalone/browser/accessibilityHelp/accessibilityHelp.js';
import * as monaco from "monaco-editor/esm/vs/editor/editor.api"
import { onMounted, ref } from "vue";
import "./work"
let editor: monaco.editor.IStandaloneCodeEditor;
const monacoEL = ref()
onMounted(()=>{
// https://www.typescriptlang.org/dev/typescript-vfs/
editor = monaco.editor.create(monacoEL.value, {
value: "function hello() {\n\talert('Hello world!');\n}",
language: 'javascript',
automaticLayout: true,
// folding: false,
// theme: 'vs-dark',
// lineNumbers: 'off',
// minimap: {
// enabled: false
// }
});
window.onresize = () => {
console.log('Window resize');
editor.layout({} as monaco.editor.IDimension);
}
})
</script>
<style scoped lang="less">
.ui-monaco-editor{
width: 100%;
height: 500px;
}
</style>

20
src/pagesUI/editor/register.ts

@ -0,0 +1,20 @@
import * as monaco from "monaco-editor";
//格式化語言
export function registerPlugins() {
// https://www.javaroad.cn/questions/156725
const cssFormatProvider = {
provideDocumentFormattingEdits(
model: monaco.editor.ITextModel,
options: any,
token: any) {
return [{
text: model.getValue() + "222222", // put formatted text here
range: model.getFullModelRange()
}];
}
};
monaco.languages.registerDocumentFormattingEditProvider('css', cssFormatProvider);
monaco.languages.registerDocumentFormattingEditProvider('javascript', cssFormatProvider);
}

25
src/pagesUI/editor/work.ts

@ -0,0 +1,25 @@
//https://stackoverflow.com/questions/65953675/import-monaco-editor-using-vite-2
import editorWorker from 'monaco-editor/esm/vs/editor/editor.worker?worker'
import jsonWorker from 'monaco-editor/esm/vs/language/json/json.worker?worker'
import cssWorker from 'monaco-editor/esm/vs/language/css/css.worker?worker'
import htmlWorker from 'monaco-editor/esm/vs/language/html/html.worker?worker'
import tsWorker from 'monaco-editor/esm/vs/language/typescript/ts.worker?worker'
//@ts-ignore
window.MonacoEnvironment = {
//@ts-ignore
getWorker(_, label) {
if (label === 'json') {
return new jsonWorker()
}
if (label === 'css' || label === 'scss' || label === 'less') {
return new cssWorker()
}
if (label === 'html' || label === 'handlebars' || label === 'razor') {
return new htmlWorker()
}
if (label === 'typescript' || label === 'javascript') {
return new tsWorker()
}
return new editorWorker()
}
}

88
src/pagesUI/verifyCodeBtn/verifyCodeBtn.vue

@ -0,0 +1,88 @@
<template>
<a-button
@click="onClick"
:disabled="isDisabled"
:loading="isLoading"
type="primary"
size="mini"
>
{{ text }}
</a-button>
</template>
<script lang="ts" setup>
import { onBeforeUnmount, ref, watchEffect } from 'vue'
const props = withDefaults(
defineProps<{
duration?: number
initText?: string
runText?: string
loadingText?: string
resetText?: string
}>(),
{
runText: '{%s}s 后重新发送',
initText: '获取验证码',
loadingText: '正在发送',
resetText: '重新获取',
duration: 60,
}
)
const emits = defineEmits<{
(event: 'update:modelValue', show: boolean): void
(event: 'send', start: () => void, done: (isDone: boolean) => void): void
}>()
const text = ref(props.initText)
const isDisabled = ref(false)
const isLoading = ref(false)
let number = props.duration
let timeID: any
onBeforeUnmount(() => {
stop()
})
function stop() {
clearInterval(timeID)
number = props.duration
text.value = props.resetText
isLoading.value = false
isDisabled.value = false
emits('update:modelValue', false)
}
//
function getText(second: string | number): string {
return props.runText.replace(/\{([^{]*?)%s(.*?)\}/g, String(second))
}
function run() {
isLoading.value = false
text.value = getText(number)
clearInterval(timeID)
timeID = setInterval(() => {
number--
text.value = getText(number)
if (number <= 0) {
stop()
}
}, 1000)
}
function onClick() {
emits(
'send',
() => {
isDisabled.value = true
isLoading.value = true
text.value = props.loadingText
},
(isDone: boolean) => {
if (isDone) {
run()
} else {
stop()
}
}
)
}
</script>

5
src/plugins/index.md

@ -0,0 +1,5 @@
https://github.com/anncwb/vite-plugin-mock/blob/main/README.zh_CN.md
https://github.com/nuysoft/Mock
https://github.com/robinvdvleuten/vuex-persistedstate
https://github.com/nuysoft/Mock/wiki/Getting-Started

13
src/plugins/index.ts

@ -0,0 +1,13 @@
import { App, InjectionKey, readonly } from "vue"
declare module 'vue' {
interface ComponentCustomProperties {
$version: string;
}
}
export default {
install(app: App) {
app.config.globalProperties.$version = '0.0.1'
},
};

95
src/router/basic/index.ts

@ -0,0 +1,95 @@
import {
PAGE_NOT_FOUND_NAME,
LAYOUT,
EXCEPTION_COMPONENT,
LOGIN_COMPONENT,
REDIRECT_NAME,
REDIRECT_COMPONENT,
} from '@/router/constant'
import { RouteRecordRaw } from 'vue-router'
let routes:RouteRecordRaw[] = []
// 读取route下的路由
const modules1 = import.meta.globEager('./route/*.ts');
// 直接从页面目录读取路由
const modules2 = import.meta.globEager('../../pages/**/*Route.ts')
console.log(modules2);
function readRoute(modules: Record<string, {
[key: string]: any;
}>) {
for (const path in modules) {
const mod = modules[path]
const module = mod.default || mod
if(Array.isArray(module)){
routes = routes.concat(module);
}else{
routes.push(module)
}
}
}
readRoute(modules1)
readRoute(modules2)
export const asyncRoute = routes
export const LOGIN_ROUTE = {
path: '/login',
name: 'Login',
hidden: true,
component: LOGIN_COMPONENT,
meta: {
title: '登录',
hideBreadcrumb: true,
hideSliderMenu: true,
},
}
// 404 on a page
export const PAGE_NOT_FOUND_ROUTE = {
path: '/:path(.*)*',
name: PAGE_NOT_FOUND_NAME,
component: LAYOUT,
children: [
{
path: '',
name: PAGE_NOT_FOUND_NAME,
component: EXCEPTION_COMPONENT,
meta: {
title: 'ErrorPage',
hideBreadcrumb: true,
hideSliderMenu: true,
},
},
],
}
export const REDIRECT_ROUTE = {
path: '/redirect',
component: LAYOUT,
name: 'RedirectTo',
hidden: true,
meta: {
title: REDIRECT_NAME,
hideBreadcrumb: true,
hideSliderMenu: true,
},
children: [
{
path: '/redirect/:path(.*)',
name: REDIRECT_NAME,
component: REDIRECT_COMPONENT,
meta: {
title: REDIRECT_NAME,
hideBreadcrumb: true,
hideSliderMenu: true,
},
},
],
}
export default [
REDIRECT_ROUTE,
LOGIN_ROUTE,
PAGE_NOT_FOUND_ROUTE,
]

0
src/router/basic/route/.gitkeep

12
src/router/constant.ts

@ -0,0 +1,12 @@
export const REDIRECT_NAME = 'Redirect';
export const PARENT_LAYOUT_NAME = 'ParentLayout';
export const PAGE_NOT_FOUND_NAME = 'PageNotFound';
export const LOGIN_COMPONENT = () => import('@/pagesSys/Login/Login.vue');
export const EXCEPTION_COMPONENT = () => import('@/pagesSys/Exception/Exception.vue');
export const REDIRECT_COMPONENT = () => import('@/pagesSys/Redirect/Redirect.vue');
/**
* @description: default layout
*/
export const LAYOUT = () => import('@/pagesSys/Layout/Layout.vue');

30
src/router/guard/index.ts

@ -0,0 +1,30 @@
import { isNavigationFailure, Router } from 'vue-router'
import { userStore } from '@/store/module/user'
import { asyncRoute } from '@/router/basic'
const WHITE_PATH = ['/login', '/home']
export default (router: Router) => {
asyncRoute.forEach((route) => {
router.addRoute(route)
})
console.log(router.getRoutes())
router.beforeEach((to, from, next) => {
const store = userStore()
const hasToken = !!store.token
if (WHITE_PATH.includes(to.path)) {
next()
} else if (to.meta?.noAuth) {
next()
} else {
if (hasToken) {
// 判断权限
next()
} else {
next(`/login?redirect=${to.path}`)
}
}
})
router.afterEach((to, from, failure) => {})
}

20
src/router/index.ts

@ -0,0 +1,20 @@
import * as VueRouter from 'vue-router'
import routes from './basic/index'
import guard from './guard/index'
const router = VueRouter.createRouter({
history: VueRouter.createWebHashHistory(),
routes,
strict: true,
scrollBehavior(to, from, savedPosition) {
if (savedPosition) {
return savedPosition
} else {
return { top: 0 }
}
},
})
guard(router)
export default router

8
src/store/index.ts

@ -0,0 +1,8 @@
import { createPinia } from 'pinia'
import { App } from 'vue'
export default {
install(app: App) {
app.use(createPinia())
},
}

13
src/store/module/user.ts

@ -0,0 +1,13 @@
import { defineStore } from "pinia"
export const userStore = defineStore("user", {
state: () => ({
token: ''
}),
getters: {
},
actions: {
},
})

23
tsconfig.json

@ -0,0 +1,23 @@
{
"compilerOptions": {
"skipLibCheck": true,
"target": "esnext",
"useDefineForClassFields": true,
"module": "esnext",
"moduleResolution": "node",
"strict": true,
"jsx": "preserve",
"sourceMap": true,
"resolveJsonModule": true,
"esModuleInterop": true,
"lib": ["esnext", "dom"],
"baseUrl": ".",
"paths": {
"@/*": ["./src/*"],
"@util": ["./src/assets/script/util"],
"#/*": ["./types/*"],
}
},
"include": ["src/**/*.ts", "src/**/*.d.ts", "src/**/*.tsx", "src/**/*.vue", "types/**/*.d.ts", "types/**/*.ts", "vite.config.ts"],
"exclude": ["node_modules"]
}

0
types/index.d.ts

37
vite.config.ts

@ -0,0 +1,37 @@
import { defineConfig, loadEnv } from 'vite'
import path from 'path'
import vue from '@vitejs/plugin-vue'
import Components from 'unplugin-vue-components/vite'
import { viteMockServe } from 'vite-plugin-mock'
import WindiCSS from 'vite-plugin-windicss'
// https://vitejs.dev/config/
export default ({ mode, command } : { mode: string, command: string}) => {
const config = <ImportMetaEnv>loadEnv(mode, process.cwd())
return defineConfig({
base: './',
resolve: {
extensions: ['.mjs', '.js', '.ts', '.jsx', '.tsx', '.json'],
alias: {
'@': path.resolve('./src'),
'@util': path.resolve('./src/assets/script/util'),
},
},
css: {
preprocessorOptions: {
less: {
additionalData: `@import (reference) "@/assets/style/global.less";`
},
},
},
plugins: [
vue(), WindiCSS(),
Components({
dts: true
}),
viteMockServe({
mockPath: 'mock'
}),
]
})
}

3
windi.config.ts

@ -0,0 +1,3 @@
import { defineConfig } from 'vite-plugin-windicss'
export default defineConfig({})
Loading…
Cancel
Save