commit
566d2223ac
27 changed files with 1165 additions and 0 deletions
@ -0,0 +1,52 @@ |
|||||
|
from flask import Flask |
||||
|
from pymongo import MongoClient |
||||
|
|
||||
|
app = Flask(__name__) |
||||
|
client = MongoClient("mongodb://localhost:27019/") |
||||
|
db = client["back"] |
||||
|
collection = db["users"] |
||||
|
|
||||
|
|
||||
|
# 测试 |
||||
|
@app.route("/") |
||||
|
def hello(): |
||||
|
return "Hello World!" |
||||
|
|
||||
|
|
||||
|
# 新增用户 |
||||
|
@app.route("/insert") |
||||
|
def insert_data(): |
||||
|
user = {"name": "John Doe", "age": 25, "city": "New York"} |
||||
|
collection.insert_one(user) |
||||
|
return "Data inserted successfully!" |
||||
|
|
||||
|
|
||||
|
# 查询用户 |
||||
|
@app.route("/query") |
||||
|
def query_data(): |
||||
|
users = collection.find() |
||||
|
result = "" |
||||
|
for user in users: |
||||
|
result += f"Name: {user['name']}, Age: {user['age']}, City: {user['city']}<br>" |
||||
|
return result |
||||
|
|
||||
|
|
||||
|
# 更新用户 |
||||
|
@app.route("/update") |
||||
|
def update_data(): |
||||
|
query = {"name": "John Doe"} |
||||
|
new_data = {"$set": {"age": 30, "city": "San Francisco"}} |
||||
|
collection.update_one(query, new_data) |
||||
|
return "Data updated successfully!" |
||||
|
|
||||
|
|
||||
|
# 删除用户 |
||||
|
@app.route("/delete") |
||||
|
def delete_data(): |
||||
|
query = {"name": "John Doe"} |
||||
|
collection.delete_one(query) |
||||
|
return "Data deleted successfully!" |
||||
|
|
||||
|
|
||||
|
if __name__ == "__main__": |
||||
|
app.run() |
@ -0,0 +1,25 @@ |
|||||
|
# Logs |
||||
|
logs |
||||
|
*.log |
||||
|
npm-debug.log* |
||||
|
yarn-debug.log* |
||||
|
yarn-error.log* |
||||
|
pnpm-debug.log* |
||||
|
lerna-debug.log* |
||||
|
|
||||
|
node_modules |
||||
|
dist |
||||
|
package-lock.json |
||||
|
dist-ssr |
||||
|
*.local |
||||
|
|
||||
|
# Editor directories and files |
||||
|
.vscode/* |
||||
|
!.vscode/extensions.json |
||||
|
.idea |
||||
|
.DS_Store |
||||
|
*.suo |
||||
|
*.ntvs* |
||||
|
*.njsproj |
||||
|
*.sln |
||||
|
*.sw? |
@ -0,0 +1,7 @@ |
|||||
|
# base |
||||
|
|
||||
|
This template should help get you started developing with Vue 3 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 |
||||
|
|
||||
|
- [VS Code](https://code.visualstudio.com/) + [Volar](https://marketplace.visualstudio.com/items?itemName=Vue.volar) (and disable Vetur) + [TypeScript Vue Plugin (Volar)](https://marketplace.visualstudio.com/items?itemName=Vue.vscode-typescript-vue-plugin). |
@ -0,0 +1,12 @@ |
|||||
|
<!DOCTYPE html> |
||||
|
<html lang="zh"> |
||||
|
<head> |
||||
|
<meta charset="UTF-8" /> |
||||
|
<link rel="icon" type="image/svg+xml" href="/vite.svg" /> |
||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" /> |
||||
|
</head> |
||||
|
<body> |
||||
|
<div id="app"></div> |
||||
|
<script type="module" src="/src/main.js"></script> |
||||
|
</body> |
||||
|
</html> |
@ -0,0 +1,30 @@ |
|||||
|
{ |
||||
|
"name": "base", |
||||
|
"private": true, |
||||
|
"version": "0.0.0", |
||||
|
"type": "module", |
||||
|
"scripts": { |
||||
|
"dev": "vite", |
||||
|
"build": "vite build", |
||||
|
"preview": "vite preview" |
||||
|
}, |
||||
|
"dependencies": { |
||||
|
"@element-plus/icons-vue": "^2.3.1", |
||||
|
"axios": "^1.6.6", |
||||
|
"dayjs": "^1.11.10", |
||||
|
"element-plus": "^2.5.3", |
||||
|
"lodash": "^4.17.21", |
||||
|
"md5": "^2.3.0", |
||||
|
"nprogress": "^0.2.0", |
||||
|
"papaparse": "^5.4.1", |
||||
|
"qs": "^6.11.2", |
||||
|
"unplugin-auto-import": "^0.17.4", |
||||
|
"unplugin-vue-components": "^0.26.0", |
||||
|
"vue": "^3.3.11", |
||||
|
"vue-router": "^4.2.5" |
||||
|
}, |
||||
|
"devDependencies": { |
||||
|
"@vitejs/plugin-vue": "^4.6.2", |
||||
|
"vite": "^5.0.8" |
||||
|
} |
||||
|
} |
@ -0,0 +1,9 @@ |
|||||
|
{ |
||||
|
"title": "可视化预警平台", |
||||
|
"jianceduixiang1path": "@/assets/img/jianceduixiang1.gif", |
||||
|
"jianceduixiang2path": "@/assets/img/jianceduixiang2.jpg", |
||||
|
"jianceduixiang3path": "@/assets/img/jianceduixiang3.png", |
||||
|
"jianceduixiang4path": "@/assets/img/jianceduixiang4.png", |
||||
|
"jianceduixiang5path": "@/assets/img/jianceduixiang5.gif", |
||||
|
"jianceduixiang6path": "@/assets/img/jianceduixiang6.gif" |
||||
|
} |
After Width: | Height: | Size: 702 B |
@ -0,0 +1,20 @@ |
|||||
|
<template> |
||||
|
<router-view /> |
||||
|
</template> |
||||
|
|
||||
|
<script> |
||||
|
|
||||
|
export default { |
||||
|
name: "app", |
||||
|
data() { |
||||
|
return {}; |
||||
|
}, |
||||
|
async mounted() { }, |
||||
|
methods: { |
||||
|
|
||||
|
}, |
||||
|
watch: {}, |
||||
|
computed: {}, |
||||
|
}; |
||||
|
</script> |
||||
|
<style scoped></style> |
@ -0,0 +1,11 @@ |
|||||
|
import _axios from "@/plugins/axios"; |
||||
|
|
||||
|
//登录、注册
|
||||
|
export function loginFun(data) { |
||||
|
return _axios({ |
||||
|
url: `/v1/login`, |
||||
|
method: "POST", |
||||
|
data, |
||||
|
}); |
||||
|
} |
||||
|
|
@ -0,0 +1,36 @@ |
|||||
|
import _axios from "@/plugins/axios"; |
||||
|
|
||||
|
//增加用户
|
||||
|
export function addUser(data) { |
||||
|
return _axios({ |
||||
|
url: `/v1/addUser`, |
||||
|
method: "POST", |
||||
|
data, |
||||
|
}); |
||||
|
} |
||||
|
|
||||
|
//删除用户
|
||||
|
export function delUser(data) { |
||||
|
return _axios({ |
||||
|
url: `/v1/delUser`, |
||||
|
method: "POST", |
||||
|
data, |
||||
|
}); |
||||
|
} |
||||
|
//修改用户
|
||||
|
export function getUser(data) { |
||||
|
return _axios({ |
||||
|
url: `/v1/updateUser`, |
||||
|
method: "POST", |
||||
|
data, |
||||
|
}); |
||||
|
} |
||||
|
//查找用户
|
||||
|
export function getUser(data) { |
||||
|
return _axios({ |
||||
|
url: `/v1/getUser`, |
||||
|
method: "POST", |
||||
|
data, |
||||
|
}); |
||||
|
} |
||||
|
|
@ -0,0 +1,3 @@ |
|||||
|
body { |
||||
|
margin: 0; |
||||
|
} |
After Width: | Height: | Size: 16 KiB |
After Width: | Height: | Size: 25 KiB |
After Width: | Height: | Size: 6.4 KiB |
After Width: | Height: | Size: 41 KiB |
After Width: | Height: | Size: 42 KiB |
@ -0,0 +1,71 @@ |
|||||
|
<template> |
||||
|
<el-form :inline="true" :disabled="formDisabled" :model="formData" label-suffix=":" label-width="100px"> |
||||
|
<el-row> |
||||
|
<el-col v-for="(headerItem, headerIndex) in formHeader" :key="headerIndex" :span="8"> |
||||
|
<el-form-item :label="headerItem.label" v-if="!(['time'].indexOf(headerItem.type) !== -1 && type === 'add')"> |
||||
|
<template v-if="['select'].indexOf(headerItem.type) !== -1"> |
||||
|
<el-select-v2 filterable v-model="formData[headerItem.inputValue]" multiple collapse-tags |
||||
|
collapse-tags-tooltip :options="headerItem.options" :placeholder="headerItem.label" style="width: 330px" |
||||
|
:max-collapse-tags="2" /> |
||||
|
</template> |
||||
|
<template v-else-if="['time'].indexOf(headerItem.type) !== -1"> |
||||
|
<el-date-picker v-model="formData[headerItem.inputValue]" type="datetime" :disabled="headerItem.disabled" |
||||
|
:placeholder="headerItem.label" format="YYYY/MM/DD HH:mm:ss" style="width: 330px" /> |
||||
|
</template> |
||||
|
<template v-else-if="['switch'].indexOf(headerItem.type) !== -1"> |
||||
|
<el-switch v-model="formData[headerItem.inputValue]" style="width: 330px"/> |
||||
|
</template> |
||||
|
<template v-else-if="['text', 'email', 'textarea', 'password'].indexOf( |
||||
|
headerItem.type |
||||
|
) !== -1 |
||||
|
"> |
||||
|
<el-input v-model="formData[headerItem.inputValue]" :placeholder="headerItem.label" :type="headerItem.type" |
||||
|
clearable :show-password="headerItem.type === 'password'" style="width: 330px" /> |
||||
|
</template> |
||||
|
</el-form-item> |
||||
|
</el-col> |
||||
|
</el-row> |
||||
|
</el-form> |
||||
|
</template> |
||||
|
<script> |
||||
|
export default { |
||||
|
name: "bug", |
||||
|
props: { |
||||
|
type: { |
||||
|
typeof: String, |
||||
|
default: () => { |
||||
|
return ""; |
||||
|
}, |
||||
|
}, |
||||
|
formHeader: { |
||||
|
typeof: Array, |
||||
|
default: () => { |
||||
|
return []; |
||||
|
}, |
||||
|
}, |
||||
|
formData: { |
||||
|
typeof: Object, |
||||
|
default: () => { |
||||
|
return {}; |
||||
|
}, |
||||
|
}, |
||||
|
formDisabled: { |
||||
|
typeof: Boolean, |
||||
|
default: () => { |
||||
|
return false; |
||||
|
}, |
||||
|
}, |
||||
|
}, |
||||
|
components: {}, |
||||
|
data() { |
||||
|
return {}; |
||||
|
}, |
||||
|
watch: {}, |
||||
|
computed: {}, |
||||
|
async mounted() { }, |
||||
|
beforeUnmount() { }, |
||||
|
|
||||
|
methods: {}, |
||||
|
}; |
||||
|
</script> |
||||
|
<style scoped></style> |
@ -0,0 +1,114 @@ |
|||||
|
<template> |
||||
|
<el-table :ref="refName" :data="tableData" class="mainTable" show-overflow-tooltip> |
||||
|
<el-table-column type="selection" width="55"> </el-table-column> |
||||
|
<el-table-column type="index" width="50"> |
||||
|
<template #default="scope"> |
||||
|
{{ scope.$index + 1 + pageSize * (currentPage - 1) }} |
||||
|
</template> |
||||
|
</el-table-column> |
||||
|
<el-table-column v-for="(headerItem, headerIndex) in tableHeader" :key="headerItem._id" |
||||
|
:min-width="headerItem.minWidth" :label="headerItem.label"> |
||||
|
<template #default="scope"> |
||||
|
<template v-if="headerItem.type === 'time'"> |
||||
|
{{ $dayjs(scope.row[headerItem.prop]).format("YYYY-MM-DD HH:mm:ss") }} |
||||
|
</template> |
||||
|
<template v-else-if="headerItem.type === 'array'"> |
||||
|
{{ $_.join(scope.row[headerItem.prop], ",") }} |
||||
|
</template> |
||||
|
<template v-else-if="headerItem.type === 'password'"> ****** </template> |
||||
|
<template v-else> |
||||
|
{{ scope.row[headerItem.prop] }} |
||||
|
</template> |
||||
|
</template> |
||||
|
</el-table-column> |
||||
|
<el-table-column fixed="right" align="center" label="Operations" width="110"> |
||||
|
<template #default="scope"> |
||||
|
<el-button link type="primary" text @click="detailInfo(scope.row)"> |
||||
|
<el-icon> |
||||
|
<InfoFilled /> |
||||
|
</el-icon> |
||||
|
</el-button> |
||||
|
<el-button link type="primary" text @click="editInfo(scope.row)"> |
||||
|
<el-icon> |
||||
|
<Edit /> |
||||
|
</el-icon> |
||||
|
</el-button> |
||||
|
</template> |
||||
|
</el-table-column> |
||||
|
</el-table> |
||||
|
<el-pagination :current-page="currentPage" :page-size="pageSize" :page-sizes="pageSizes" |
||||
|
layout="total, sizes, prev, pager, next, jumper" :total="total" @size-change="handleSizeChange" |
||||
|
@current-change="handleCurrentChange" class="pagination" /> |
||||
|
</template> |
||||
|
<script> |
||||
|
export default { |
||||
|
name: "app", |
||||
|
emits: ["detailInfo", "editInfo", "handleSizeChange", "handleCurrentChange"], |
||||
|
props: { |
||||
|
refName: { |
||||
|
typeof: String, |
||||
|
default: "table", |
||||
|
}, |
||||
|
tableHeader: { |
||||
|
typeof: Array, |
||||
|
default: () => { |
||||
|
return []; |
||||
|
}, |
||||
|
}, |
||||
|
tableData: { |
||||
|
typeof: Array, |
||||
|
default: () => { |
||||
|
return []; |
||||
|
}, |
||||
|
}, |
||||
|
currentPage: { |
||||
|
typeof: Number, |
||||
|
default: 1, |
||||
|
}, |
||||
|
pageSize: { |
||||
|
typeof: Number, |
||||
|
default: 20, |
||||
|
}, |
||||
|
pageSizes: { |
||||
|
typeof: Number, |
||||
|
default: () => { |
||||
|
return [16, 50, 100, 200]; |
||||
|
}, |
||||
|
}, |
||||
|
total: { |
||||
|
typeof: Number, |
||||
|
default: 0, |
||||
|
}, |
||||
|
}, |
||||
|
data() { |
||||
|
return {}; |
||||
|
}, |
||||
|
async mounted() { }, |
||||
|
methods: { |
||||
|
detailInfo(item) { |
||||
|
this.$emit("detailInfo", item); |
||||
|
}, |
||||
|
editInfo(item) { |
||||
|
this.$emit("editInfo", item); |
||||
|
}, |
||||
|
handleSizeChange(sizeVal) { |
||||
|
this.$emit("handleSizeChange", sizeVal); |
||||
|
}, |
||||
|
handleCurrentChange(currentSize) { |
||||
|
this.$emit("handleCurrentChange", currentSize); |
||||
|
}, |
||||
|
}, |
||||
|
watch: {}, |
||||
|
computed: {}, |
||||
|
}; |
||||
|
</script> |
||||
|
<style scoped> |
||||
|
.mainTable { |
||||
|
height: calc(100vh - 208px); |
||||
|
width: 100%; |
||||
|
} |
||||
|
|
||||
|
.pagination { |
||||
|
justify-content: right; |
||||
|
} |
||||
|
</style> |
@ -0,0 +1,22 @@ |
|||||
|
import App from "./App.vue"; |
||||
|
import { createApp } from "vue"; |
||||
|
// 引入路由实例
|
||||
|
import router from "@/router/index.js"; |
||||
|
//全局引入lodash
|
||||
|
import _ from "lodash"; |
||||
|
//全局引入lodash
|
||||
|
import dayjs from "dayjs"; |
||||
|
// 引入css
|
||||
|
import "@/assets/css/base.css"; |
||||
|
//全局引入element-plus所有图标
|
||||
|
import * as ElementPlusIconsVue from '@element-plus/icons-vue' |
||||
|
|
||||
|
const app = createApp(App); |
||||
|
for (const [key, component] of Object.entries(ElementPlusIconsVue)) { |
||||
|
app.component(key, component) |
||||
|
} |
||||
|
app.config.globalProperties.$_ = _; //挂载到app实例上
|
||||
|
app.config.globalProperties.$dayjs = dayjs; //挂载到app实例上
|
||||
|
app |
||||
|
.use(router) //路由
|
||||
|
app.mount("#app"); |
@ -0,0 +1,40 @@ |
|||||
|
import axios from "axios"; |
||||
|
import qs from "qs"; |
||||
|
import router from "@/router/index.js"; |
||||
|
let config = { |
||||
|
baseURL: "", |
||||
|
timeout: 600 * 1000, |
||||
|
withCredentials: true, // Check cross-site Access-Control
|
||||
|
/* `paramsSerializer` 是一个负责 `params` 序列化的函数 |
||||
|
qs.stringify({ a: ['b', 'c'] }, { arrayFormat: 'indices' }) |
||||
|
// 'a[0]=b&a[1]=c'
|
||||
|
qs.stringify({ a: ['b', 'c'] }, { arrayFormat: 'brackets' }) |
||||
|
// 'a[]=b&a[]=c'
|
||||
|
qs.stringify({ a: ['b', 'c'] }, { arrayFormat: 'repeat' }) |
||||
|
// 'a=b&a=c'
|
||||
|
*/ |
||||
|
paramsSerializer: (params) => { |
||||
|
return qs.stringify(params, { arrayFormat: "indices" }); |
||||
|
}, |
||||
|
}; |
||||
|
|
||||
|
const _axios = axios.create(config); |
||||
|
|
||||
|
_axios.interceptors.request.use( |
||||
|
(config) => { |
||||
|
return config; |
||||
|
}, |
||||
|
(error) => { |
||||
|
return Promise.reject(error); |
||||
|
} |
||||
|
); |
||||
|
_axios.interceptors.response.use( |
||||
|
(response) => { |
||||
|
return Promise.resolve(response.data); |
||||
|
}, |
||||
|
(error) => { |
||||
|
ElMessage.error(error.response.data.message); |
||||
|
return Promise.reject(error); |
||||
|
} |
||||
|
); |
||||
|
export default _axios; |
@ -0,0 +1,44 @@ |
|||||
|
import { createRouter, createWebHistory } from "vue-router"; |
||||
|
import { close, start } from "@/utils/nprogress"; |
||||
|
import home from "../views/home.vue" |
||||
|
const routes = [ |
||||
|
{ |
||||
|
path: "/", |
||||
|
name: "home", |
||||
|
component: home, |
||||
|
redirect: "/bug", |
||||
|
children: [{ |
||||
|
path: "bug", |
||||
|
name: "bug", |
||||
|
component: () => import("../views/bug.vue"), |
||||
|
}, |
||||
|
{ |
||||
|
path: "user", |
||||
|
name: "user", |
||||
|
component: () => import("../views/user.vue"), |
||||
|
},] |
||||
|
}, |
||||
|
{ |
||||
|
path: "/login", |
||||
|
name: "login", |
||||
|
component: () => import("../views/login.vue"), |
||||
|
}, |
||||
|
{ name: "重定向", path: "/:catchAll(.*)", redirect: "/bug" }, |
||||
|
]; |
||||
|
|
||||
|
// 创建路由实例
|
||||
|
const router = createRouter({ |
||||
|
history: createWebHistory(), |
||||
|
routes, //路由表
|
||||
|
}); |
||||
|
let witheRouter = ["login"]; |
||||
|
router.beforeEach((pre, next) => { |
||||
|
if (witheRouter.indexOf(pre.name) !== -1) { |
||||
|
} |
||||
|
start(); |
||||
|
}); |
||||
|
|
||||
|
router.afterEach(() => { |
||||
|
close(); |
||||
|
}); |
||||
|
export default router; |
@ -0,0 +1,79 @@ |
|||||
|
:root { |
||||
|
font-family: Inter, system-ui, Avenir, Helvetica, Arial, sans-serif; |
||||
|
line-height: 1.5; |
||||
|
font-weight: 400; |
||||
|
|
||||
|
color-scheme: light dark; |
||||
|
color: rgba(255, 255, 255, 0.87); |
||||
|
background-color: #242424; |
||||
|
|
||||
|
font-synthesis: none; |
||||
|
text-rendering: optimizeLegibility; |
||||
|
-webkit-font-smoothing: antialiased; |
||||
|
-moz-osx-font-smoothing: grayscale; |
||||
|
} |
||||
|
|
||||
|
a { |
||||
|
font-weight: 500; |
||||
|
color: #646cff; |
||||
|
text-decoration: inherit; |
||||
|
} |
||||
|
a:hover { |
||||
|
color: #535bf2; |
||||
|
} |
||||
|
|
||||
|
body { |
||||
|
margin: 0; |
||||
|
display: flex; |
||||
|
place-items: center; |
||||
|
min-width: 320px; |
||||
|
min-height: 100vh; |
||||
|
} |
||||
|
|
||||
|
h1 { |
||||
|
font-size: 3.2em; |
||||
|
line-height: 1.1; |
||||
|
} |
||||
|
|
||||
|
button { |
||||
|
border-radius: 8px; |
||||
|
border: 1px solid transparent; |
||||
|
padding: 0.6em 1.2em; |
||||
|
font-size: 1em; |
||||
|
font-weight: 500; |
||||
|
font-family: inherit; |
||||
|
background-color: #1a1a1a; |
||||
|
cursor: pointer; |
||||
|
transition: border-color 0.25s; |
||||
|
} |
||||
|
button:hover { |
||||
|
border-color: #646cff; |
||||
|
} |
||||
|
button:focus, |
||||
|
button:focus-visible { |
||||
|
outline: 4px auto -webkit-focus-ring-color; |
||||
|
} |
||||
|
|
||||
|
.card { |
||||
|
padding: 2em; |
||||
|
} |
||||
|
|
||||
|
#app { |
||||
|
max-width: 1280px; |
||||
|
margin: 0 auto; |
||||
|
padding: 2rem; |
||||
|
text-align: center; |
||||
|
} |
||||
|
|
||||
|
@media (prefers-color-scheme: light) { |
||||
|
:root { |
||||
|
color: #213547; |
||||
|
background-color: #ffffff; |
||||
|
} |
||||
|
a:hover { |
||||
|
color: #747bff; |
||||
|
} |
||||
|
button { |
||||
|
background-color: #f9f9f9; |
||||
|
} |
||||
|
} |
@ -0,0 +1,19 @@ |
|||||
|
import NProgress from 'nprogress' |
||||
|
import 'nprogress/nprogress.css' |
||||
|
NProgress.configure({ |
||||
|
easing: 'ease', // 动画方式
|
||||
|
speed: 1000, // 递增进度条的速度
|
||||
|
showSpinner: false, // 是否显示加载ico
|
||||
|
trickleSpeed: 200, // 自动递增间隔
|
||||
|
minimum: 0.3, // 更改启动时使用的最小百分比
|
||||
|
parent: 'body', //指定进度条的父容器
|
||||
|
}) |
||||
|
// 打开进度条
|
||||
|
export const start = () => { |
||||
|
NProgress.start() |
||||
|
} |
||||
|
|
||||
|
// 关闭进度条
|
||||
|
export const close = () => { |
||||
|
NProgress.done() |
||||
|
} |
@ -0,0 +1,85 @@ |
|||||
|
<template> |
||||
|
<el-container> |
||||
|
<el-header class="appHeader"> |
||||
|
<el-row> |
||||
|
<el-col :span="23" style="align-self: center"> |
||||
|
<el-image style="width: 60px; height: 60px" :src="logoUrl" fit="contain" /> |
||||
|
</el-col> |
||||
|
<el-col :span="1" style="align-self: center"> |
||||
|
<el-dropdown> |
||||
|
<span> <el-avatar :src="teacherUrl" /> </span> |
||||
|
<template #dropdown> |
||||
|
<el-dropdown-menu> |
||||
|
<el-dropdown-item @click="logout">退出</el-dropdown-item> |
||||
|
</el-dropdown-menu> |
||||
|
</template> |
||||
|
</el-dropdown> |
||||
|
</el-col> |
||||
|
</el-row> |
||||
|
</el-header> |
||||
|
<el-container> |
||||
|
<el-container> |
||||
|
<el-main class="appMain"> |
||||
|
<router-view /> |
||||
|
</el-main> |
||||
|
<el-footer class="appFooter">版权所有</el-footer> |
||||
|
</el-container> |
||||
|
</el-container> |
||||
|
</el-container> |
||||
|
</template> |
||||
|
|
||||
|
<script> |
||||
|
import logo from "@/assets/img/logo.jpg"; |
||||
|
import teacher from "@/assets/img/teacher.png"; |
||||
|
import student from "@/assets/img/student.png"; |
||||
|
export default { |
||||
|
name: "home", |
||||
|
data() { |
||||
|
return { |
||||
|
logoUrl: logo, |
||||
|
teacherUrl: teacher, |
||||
|
studentUrl: student, |
||||
|
userName: "", |
||||
|
}; |
||||
|
}, |
||||
|
async mounted() { |
||||
|
await this.init(); |
||||
|
}, |
||||
|
methods: { |
||||
|
async init() { |
||||
|
if (!localStorage.getItem("userId")) { |
||||
|
this.$router.push("/login"); |
||||
|
} |
||||
|
this.userName = localStorage.getItem("userName") |
||||
|
}, |
||||
|
async logout() { |
||||
|
localStorage.clear() |
||||
|
await this.init() |
||||
|
}, |
||||
|
routerFun(route) { |
||||
|
this.$router.push(route); |
||||
|
}, |
||||
|
}, |
||||
|
watch: {}, |
||||
|
computed: {}, |
||||
|
}; |
||||
|
</script> |
||||
|
<style scoped> |
||||
|
.appHeader { |
||||
|
background-color: #67c23a; |
||||
|
} |
||||
|
|
||||
|
.appMain { |
||||
|
height: calc(100vh - 120px); |
||||
|
padding: 8px; |
||||
|
} |
||||
|
|
||||
|
.appAside { |
||||
|
height: calc(100vh - 60px); |
||||
|
background-color: #eee; |
||||
|
} |
||||
|
|
||||
|
.appFooter { |
||||
|
background-color: #eee; |
||||
|
} |
||||
|
</style> |
@ -0,0 +1,103 @@ |
|||||
|
<template> |
||||
|
<div class="loginMain"> |
||||
|
<div class="login"> |
||||
|
<el-card shadow="hover" style="width: 450px"> |
||||
|
<template #header> |
||||
|
<div style="text-align: center"> |
||||
|
<el-button class="button" text @click="regist = false"> |
||||
|
登录 |
||||
|
</el-button> |
||||
|
<el-button class="button" text @click="regist = true"> |
||||
|
注册 |
||||
|
</el-button> |
||||
|
</div> |
||||
|
</template> |
||||
|
<div> |
||||
|
<el-form :model="regForm" label-suffix=":"> |
||||
|
<el-form-item label="账号"> |
||||
|
<el-input v-model="regForm.name" placeholder="请输入账号" /> |
||||
|
</el-form-item> |
||||
|
<el-form-item label="密码"> |
||||
|
<el-input v-model="regForm.password" type="password" placeholder="请输入密码" show-password /> |
||||
|
</el-form-item> |
||||
|
<el-form-item label="邮箱" v-if="regist"> |
||||
|
<el-input v-model="regForm.email" placeholder="请输入邮箱" /> |
||||
|
</el-form-item> |
||||
|
</el-form> |
||||
|
</div> |
||||
|
<template #footer> |
||||
|
<div style="text-align: center"> |
||||
|
<el-button class="button" v-if="regist" @click="login"> |
||||
|
注册 |
||||
|
</el-button> |
||||
|
<el-button class="button" v-else @click="login"> 登录 </el-button> |
||||
|
</div> |
||||
|
</template> |
||||
|
</el-card> |
||||
|
</div> |
||||
|
</div> |
||||
|
</template> |
||||
|
<script> |
||||
|
import { |
||||
|
loginFun, //登录 |
||||
|
} from "@/api/login"; |
||||
|
import md5 from "md5" |
||||
|
export default { |
||||
|
name: "login", |
||||
|
components: {}, |
||||
|
data() { |
||||
|
return { |
||||
|
regist: false, |
||||
|
regForm: { |
||||
|
name: "", |
||||
|
password: "", |
||||
|
email: "", |
||||
|
}, |
||||
|
}; |
||||
|
}, |
||||
|
watch: {}, |
||||
|
computed: {}, |
||||
|
async mounted() { |
||||
|
await this.init(); |
||||
|
}, |
||||
|
beforeUnmount() { }, |
||||
|
|
||||
|
methods: { |
||||
|
dealForm() { |
||||
|
return { |
||||
|
regist: this.regist, |
||||
|
name: this.regForm.name, |
||||
|
password: md5(this.regForm.password), |
||||
|
email: this.regForm.email, |
||||
|
} |
||||
|
}, |
||||
|
async init() { }, |
||||
|
async login() { |
||||
|
let formData = this.dealForm() |
||||
|
let res = await loginFun(formData) |
||||
|
if (!res.hasError) { |
||||
|
ElMessage.success("登陆成功"); |
||||
|
localStorage.setItem("userId", res.list._id) |
||||
|
localStorage.setItem("userName", res.list.name) |
||||
|
this.$router.push("/bug") |
||||
|
} else { |
||||
|
ElMessage.error(res.msg); |
||||
|
localStorage.clear() |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
}; |
||||
|
</script> |
||||
|
<style scoped> |
||||
|
.loginMain { |
||||
|
background-image: url("@/assets/img/login.jpg"); |
||||
|
background-size: 100% 100%; |
||||
|
height: 100vh; |
||||
|
width: 100vw; |
||||
|
} |
||||
|
|
||||
|
.login { |
||||
|
padding-top: calc(50vh - 165px); |
||||
|
padding-left: calc(50vw - 225px); |
||||
|
} |
||||
|
</style> |
@ -0,0 +1,340 @@ |
|||||
|
<template> |
||||
|
<el-row style="margin: 0 0 8px 0"> |
||||
|
<el-col :span="4"> |
||||
|
<el-button icon="Plus" type="primary" @click="addUserData"> |
||||
|
新增 |
||||
|
</el-button> |
||||
|
<el-button icon="Delete" type="danger" @click="delUserData"> |
||||
|
删除 |
||||
|
</el-button> |
||||
|
<el-button icon="Refresh" type="info" @click="init({})" text> |
||||
|
刷新 |
||||
|
</el-button> |
||||
|
</el-col> |
||||
|
<el-col :span="2"> |
||||
|
<el-upload |
||||
|
:limit="1" |
||||
|
accept="text/csv" |
||||
|
:auto-upload="false" |
||||
|
:on-change="fileChange" |
||||
|
> |
||||
|
<template #trigger> |
||||
|
<el-button type="primary">文件上传</el-button> |
||||
|
</template> |
||||
|
<template #file> </template> |
||||
|
</el-upload> |
||||
|
</el-col> |
||||
|
<el-col :span="6"> |
||||
|
<el-input |
||||
|
v-model="searchValue" |
||||
|
placeholder="Please input email" |
||||
|
@keyup.enter="searchId" |
||||
|
> |
||||
|
<template #append> |
||||
|
<el-button @click="searchId" icon="Search" /> |
||||
|
</template> |
||||
|
</el-input> |
||||
|
</el-col> |
||||
|
</el-row> |
||||
|
<div v-loading="loading"> |
||||
|
<vueTable |
||||
|
ref="userTableParent" |
||||
|
:refName="refName" |
||||
|
:tableHeader="tableHeader" |
||||
|
:tableData="tableData" |
||||
|
:currentPage="currentPage" |
||||
|
:pageSize="pageSize" |
||||
|
:pageSizes="pageSizes" |
||||
|
:total="total" |
||||
|
@detailInfo="detailInfo" |
||||
|
@editInfo="editInfo" |
||||
|
@handleSizeChange="handleSizeChange" |
||||
|
@handleCurrentChange="handleCurrentChange" |
||||
|
> |
||||
|
</vueTable> |
||||
|
</div> |
||||
|
|
||||
|
<el-dialog |
||||
|
v-model="addDialog.show" |
||||
|
:title="addDialog.title" |
||||
|
width="80%" |
||||
|
draggable |
||||
|
:close-on-click-modal="formDisabled" |
||||
|
> |
||||
|
<changeItem |
||||
|
:formData="formData" |
||||
|
:formDisabled="formDisabled" |
||||
|
:type="addDialog.type" |
||||
|
:formHeader="formHeaderLocal" |
||||
|
> |
||||
|
</changeItem> |
||||
|
<template #footer v-if="!formDisabled"> |
||||
|
<div class="dialog-footer"> |
||||
|
<el-button @click="closeDialog(addDialog)" type="info" plain> |
||||
|
取消 |
||||
|
</el-button> |
||||
|
<el-button type="primary" @click="submitAdd" plain> 提交 </el-button> |
||||
|
</div> |
||||
|
</template> |
||||
|
</el-dialog> |
||||
|
</template> |
||||
|
<script> |
||||
|
import md5 from "md5"; |
||||
|
import Papa from "papaparse"; |
||||
|
import vueTable from "../component/table.vue"; |
||||
|
import changeItem from "../component/changeItem.vue"; |
||||
|
import { |
||||
|
getUser, //获取user信息 |
||||
|
addUser, //新增user信息 |
||||
|
delUser, //删除user信息 |
||||
|
} from "@/api/user"; |
||||
|
export default { |
||||
|
name: "user", |
||||
|
components: { vueTable, changeItem }, |
||||
|
data() { |
||||
|
return { |
||||
|
refName: "userTable", |
||||
|
loading: false, |
||||
|
tableHeader: [ |
||||
|
{ prop: "name", label: "name" }, |
||||
|
{ prop: "password", label: "password", type: "password" }, |
||||
|
{ prop: "email", label: "email" }, |
||||
|
{ prop: "loginTime", label: "loginTime", type: "time", minWidth: 120 }, |
||||
|
{ prop: "create_at", label: "create_at", type: "time", minWidth: 120 }, |
||||
|
{ prop: "update_at", label: "update_at", type: "time", minWidth: 120 }, |
||||
|
], |
||||
|
tableData: [], |
||||
|
currentPage: 1, |
||||
|
pageSize: 16, |
||||
|
pageSizes: [16, 50, 100, 200], |
||||
|
total: 0, |
||||
|
addDialog: { |
||||
|
type: "add", |
||||
|
show: false, |
||||
|
title: "新增用户", |
||||
|
}, |
||||
|
formData: { |
||||
|
name: "", |
||||
|
password: "", |
||||
|
email: "", |
||||
|
loginTime: "", |
||||
|
}, |
||||
|
formHeader: [ |
||||
|
{ |
||||
|
type: "text", |
||||
|
label: "name", |
||||
|
inputValue: "name", |
||||
|
}, |
||||
|
{ |
||||
|
type: "password", |
||||
|
label: "password", |
||||
|
inputValue: "password", |
||||
|
}, |
||||
|
{ |
||||
|
type: "email", |
||||
|
label: "email", |
||||
|
inputValue: "email", |
||||
|
}, |
||||
|
{ |
||||
|
type: "switch", |
||||
|
label: "allowRegist", |
||||
|
inputValue: "allowRegist", |
||||
|
hidden: true, |
||||
|
}, |
||||
|
{ |
||||
|
type: "time", |
||||
|
disabled: true, |
||||
|
label: "loginTime", |
||||
|
inputValue: "loginTime", |
||||
|
}, |
||||
|
{ |
||||
|
type: "time", |
||||
|
disabled: true, |
||||
|
label: "create_at", |
||||
|
inputValue: "create_at", |
||||
|
}, |
||||
|
{ |
||||
|
type: "time", |
||||
|
disabled: true, |
||||
|
label: "update_at", |
||||
|
inputValue: "update_at", |
||||
|
}, |
||||
|
], |
||||
|
formHeaderLocal: [], |
||||
|
formDisabled: false, |
||||
|
searchValue: "", |
||||
|
}; |
||||
|
}, |
||||
|
watch: {}, |
||||
|
computed: {}, |
||||
|
async mounted() { |
||||
|
await this.init(); |
||||
|
}, |
||||
|
beforeUnmount() {}, |
||||
|
|
||||
|
methods: { |
||||
|
async init(params = {}, isReload = true) { |
||||
|
if (isReload) { |
||||
|
this.searchValue = ""; |
||||
|
} |
||||
|
this.loading = true; |
||||
|
let res = await getUser({ |
||||
|
subType: "get", |
||||
|
returnData: [ |
||||
|
"name", |
||||
|
"email", |
||||
|
"allowRegist", |
||||
|
"loginTime", |
||||
|
"create_at", |
||||
|
"update_at", |
||||
|
], |
||||
|
pageSize: this.pageSize, |
||||
|
currentPage: this.currentPage, |
||||
|
...params, |
||||
|
}); |
||||
|
this.tableData = res.list; |
||||
|
this.total = res.count; |
||||
|
this.loading = false; |
||||
|
}, |
||||
|
async addUserData() { |
||||
|
this.addDialog = { |
||||
|
type: "add", |
||||
|
show: true, |
||||
|
title: "新增用户", |
||||
|
}; |
||||
|
this.formData = {}; |
||||
|
this.formHeaderLocal = this.$_.reject(this.formHeader, { hidden: true }); |
||||
|
this.formDisabled = false; |
||||
|
}, |
||||
|
closeDialog(addDialog) { |
||||
|
addDialog.show = false; |
||||
|
}, |
||||
|
async submitAdd() { |
||||
|
let subData = { subType: this.addDialog.type }; |
||||
|
for (let i = 0; i < this.formHeader.length; i++) { |
||||
|
let elei = this.formHeader[i]; |
||||
|
if (elei.type !== "time") { |
||||
|
subData[elei.inputValue] = this.formData[elei.inputValue]; |
||||
|
} |
||||
|
if (elei.type === "select") { |
||||
|
subData[`${elei.inputValue}_str`] = this.$_.join( |
||||
|
this.formData[elei.inputValue], |
||||
|
"," |
||||
|
); |
||||
|
} |
||||
|
} |
||||
|
if (this.addDialog.type === "edit") { |
||||
|
subData["_id"] = this.formData._id; |
||||
|
} |
||||
|
subData["password"] = md5(this.$_.trim(this.formData.password)); |
||||
|
try { |
||||
|
let res = await addUser(subData); |
||||
|
ElMessage({ |
||||
|
message: res.msg, |
||||
|
type: "success", |
||||
|
}); |
||||
|
} catch (error) { |
||||
|
} finally { |
||||
|
this.closeDialog(this.addDialog); |
||||
|
await this.init(); |
||||
|
} |
||||
|
}, |
||||
|
async delUserData() { |
||||
|
let rowList = |
||||
|
this.$refs[`${this.refName}Parent`].$refs[ |
||||
|
this.refName |
||||
|
].getSelectionRows(); |
||||
|
let delData = []; |
||||
|
let delDispalyData = []; |
||||
|
for (let i = 0; i < rowList.length; i++) { |
||||
|
let elei = rowList[i]; |
||||
|
delData.push(elei._id); |
||||
|
delDispalyData.push(`${elei.BugID}`); |
||||
|
} |
||||
|
ElMessageBox.confirm(`${delDispalyData.join("、")}`, "删除", { |
||||
|
confirmButtonText: "确认", |
||||
|
cancelButtonText: "取消", |
||||
|
type: "error", |
||||
|
}).then(async () => { |
||||
|
let res = await delUser({ subType: "del", ids: delData }); |
||||
|
await this.init(); |
||||
|
ElMessage({ |
||||
|
message: res.msg, |
||||
|
type: "success", |
||||
|
}); |
||||
|
}); |
||||
|
}, |
||||
|
detailInfo(item) { |
||||
|
this.addDialog = { |
||||
|
type: "info", |
||||
|
show: true, |
||||
|
title: `查看${item.name}`, |
||||
|
}; |
||||
|
if (item.name !== "admin") { |
||||
|
this.formHeaderLocal = this.$_.reject(this.formHeader, { |
||||
|
hidden: true, |
||||
|
}); |
||||
|
} else { |
||||
|
this.formHeaderLocal = this.$_.cloneDeep(this.formHeader); |
||||
|
} |
||||
|
this.formDisabled = true; |
||||
|
this.formData = item; |
||||
|
}, |
||||
|
editInfo(item) { |
||||
|
this.addDialog = { |
||||
|
type: "edit", |
||||
|
show: true, |
||||
|
title: `编辑${item.name}`, |
||||
|
}; |
||||
|
if (item.name !== "admin") { |
||||
|
this.formHeaderLocal = this.$_.reject(this.formHeader, { |
||||
|
hidden: true, |
||||
|
}); |
||||
|
} else { |
||||
|
this.formHeaderLocal = this.$_.cloneDeep(this.formHeader); |
||||
|
} |
||||
|
this.formDisabled = false; |
||||
|
this.formData = { ...item }; |
||||
|
}, |
||||
|
async handleSizeChange(pageSize) { |
||||
|
this.pageSize = pageSize; |
||||
|
await this.init(); |
||||
|
}, |
||||
|
async handleCurrentChange(currentPage) { |
||||
|
this.currentPage = currentPage; |
||||
|
await this.init(); |
||||
|
}, |
||||
|
fileChange(file) { |
||||
|
let fileData = []; |
||||
|
let that = this; |
||||
|
Papa.parse(file.raw, { |
||||
|
complete: async (result) => { |
||||
|
for (let i = 1; i < result.data.length; i++) { |
||||
|
let elei = result.data[i]; |
||||
|
fileData.push({ |
||||
|
name: that.$_.trim(elei[4]), |
||||
|
password: md5(that.$_.trim(elei[4])), |
||||
|
email: that.$_.trim(elei[4]), |
||||
|
}); |
||||
|
} |
||||
|
let fileDataLocal = that.$_.uniqBy(fileData, "name"); |
||||
|
let res = await addUser({ subType: "multi", list: fileDataLocal }); |
||||
|
await this.init(); |
||||
|
ElMessage({ |
||||
|
message: res.msg, |
||||
|
type: "success", |
||||
|
}); |
||||
|
}, |
||||
|
}); |
||||
|
}, |
||||
|
async searchId() { |
||||
|
if (this.$_.trim(this.searchValue)) { |
||||
|
await this.init({ email: this.searchValue }, false); |
||||
|
} else { |
||||
|
await this.init(); |
||||
|
} |
||||
|
}, |
||||
|
}, |
||||
|
}; |
||||
|
</script> |
||||
|
<style scoped></style> |
@ -0,0 +1,42 @@ |
|||||
|
// Plugins
|
||||
|
import vue from "@vitejs/plugin-vue"; |
||||
|
|
||||
|
// Utilities
|
||||
|
import { defineConfig } from "vite"; |
||||
|
import { fileURLToPath, URL } from "node:url"; |
||||
|
import AutoImport from "unplugin-auto-import/vite"; |
||||
|
import Components from "unplugin-vue-components/vite"; |
||||
|
import { ElementPlusResolver } from "unplugin-vue-components/resolvers"; |
||||
|
|
||||
|
// https://vitejs.dev/config/
|
||||
|
export default defineConfig({ |
||||
|
plugins: [ |
||||
|
vue(), |
||||
|
AutoImport({ |
||||
|
resolvers: [ElementPlusResolver()], |
||||
|
}), |
||||
|
Components({ |
||||
|
resolvers: [ElementPlusResolver()], |
||||
|
}), |
||||
|
], |
||||
|
define: { "process.env": {} }, |
||||
|
resolve: { |
||||
|
alias: { |
||||
|
"@": fileURLToPath(new URL("./src", import.meta.url)), |
||||
|
}, |
||||
|
extensions: [".js", ".json", ".jsx", ".mjs", ".ts", ".tsx", ".vue"], |
||||
|
}, |
||||
|
server: { |
||||
|
port: 3000, |
||||
|
proxy: { |
||||
|
"/v1": { |
||||
|
target: "http://127.0.0.1:7001/v1", |
||||
|
changeOrigin: true, |
||||
|
rewrite: (path) => path.replace(/^\/v1/, ""), |
||||
|
}, |
||||
|
}, |
||||
|
}, |
||||
|
build: { |
||||
|
sourcemap: true, |
||||
|
}, |
||||
|
}); |
Loading…
Reference in new issue