Browse Source

手机端

2545307760@qq.com 5 months ago
commit
c947b38be5

+ 14 - 0
.env.development

@@ -0,0 +1,14 @@
+ENV = 'development'
+
+VITE_CLI_PORT = 8080
+VITE_SERVER_PORT = 8220
+VITE_BASE_API = /api
+VITE_FILE_API = /api
+VITE_BASE_PATH = http://192.168.110.218
+VITE_POSITION = close
+VITE_EDITOR = webstorm
+
+
+// VITE_EDITOR = webstorm 如果使用webstorm开发且要使用dom定位到代码行功能 请先自定添加 webstorm到环境变量 再将VITE_EDITOR值修改为webstorm
+// 如果使用docker-compose开发模式,设置为下面的地址或本机主机IP
+// VITE_BASE_PATH = http://106.52.134.22

+ 8 - 0
.env.production

@@ -0,0 +1,8 @@
+ENV = 'production'
+
+VITE_CLI_PORT = 8221
+VITE_SERVER_PORT = 443
+VITE_BASE_API = /lc_finance_api
+VITE_FILE_API = /lc_finance_api
+#下方修改为你的线上ip
+VITE_BASE_PATH = http://110.40.223.170

+ 24 - 0
.gitignore

@@ -0,0 +1,24 @@
+# Logs
+logs
+*.log
+npm-debug.log*
+yarn-debug.log*
+yarn-error.log*
+pnpm-debug.log*
+lerna-debug.log*
+
+node_modules
+dist
+dist-ssr
+*.local
+
+# Editor directories and files
+.vscode/*
+!.vscode/extensions.json
+.idea
+.DS_Store
+*.suo
+*.ntvs*
+*.njsproj
+*.sln
+*.sw?

+ 3 - 0
.vscode/extensions.json

@@ -0,0 +1,3 @@
+{
+  "recommendations": ["Vue.volar"]
+}

+ 5 - 0
README.md

@@ -0,0 +1,5 @@
+# Vue 3 + Vite
+
+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.
+
+Learn more about IDE Support for Vue in the [Vue Docs Scaling up Guide](https://vuejs.org/guide/scaling-up/tooling.html#ide-support).

+ 13 - 0
index.html

@@ -0,0 +1,13 @@
+<!doctype html>
+<html lang="en">
+  <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" />
+    <title>Vite + Vue</title>
+  </head>
+  <body>
+    <div id="app"></div>
+    <script type="module" src="/src/main.js"></script>
+  </body>
+</html>

+ 32 - 0
package.json

@@ -0,0 +1,32 @@
+{
+  "name": "oa",
+  "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.7.7",
+    "element-plus": "^2.8.4",
+    "js-cookie": "^3.0.5",
+    "jsencrypt": "^3.3.2",
+    "less": "^4.2.0",
+    "path": "^0.12.7",
+    "pinia": "^2.2.4",
+    "process": "^0.11.10",
+    "sass": "^1.79.4",
+    "vant": "^4.9.7",
+    "vue": "^3.4.37",
+    "vue-router": "^4.4.5"
+  },
+  "devDependencies": {
+    "@vitejs/plugin-vue": "^5.1.2",
+    "unplugin-auto-import": "^0.18.3",
+    "unplugin-vue-components": "^0.27.4",
+    "vite": "^5.4.1"
+  }
+}

File diff suppressed because it is too large
+ 1667 - 0
pnpm-lock.yaml


File diff suppressed because it is too large
+ 1 - 0
public/vite.svg


+ 11 - 0
src/App.vue

@@ -0,0 +1,11 @@
+<script setup>
+
+</script>
+
+<template>
+  <router-view />
+</template>
+
+<style scoped>
+
+</style>

+ 26 - 0
src/api/login.js

@@ -0,0 +1,26 @@
+import service from "../utils/request.js";
+
+// 验证码
+export const captcha = () => {
+    return service({
+        url: '/base/captcha',
+        method: 'post'
+    })
+}
+
+// 登录
+export const register = (data) => {
+    return service({
+        url: '/base/login',
+        method: 'post',
+        data
+    })
+}
+
+// 返回公钥
+export const routerPublicKey = () => {
+    return service({
+        url: '/base/routerPublicKey',
+        method: 'get'
+    })
+}

+ 8 - 0
src/api/menu.js

@@ -0,0 +1,8 @@
+import service from "../utils/request.js";
+// Menu列表
+export const asyncMenu = () => {
+    return service({
+        url: '/menu/getMenu',
+        method: 'post'
+    })
+}

+ 35 - 0
src/api/project.js

@@ -0,0 +1,35 @@
+import service from '@/utils/request'
+
+// 查询所有项目
+export const getAllProject = (data) => {
+    return service({
+        url: '/project/queryAllProject',
+        method: 'get'
+    })
+}
+
+// 查询项目文件
+export const getProjectFile = (data) => {
+    return service({
+        url: '/project/queryFileList',
+        method: 'post',
+        data
+    })
+}
+
+// 查询文件类型
+export const getFileType = () => {
+    return service({
+        url: '/file/queryFileGenre',
+        method: 'get'
+    })
+}
+
+// 项目文件上传
+export const createProjectFile = (data) => {
+    return service({
+        url: '/project/createProjectFile',
+        method: 'post',
+        data: data
+    })
+}

+ 8 - 0
src/api/role.js

@@ -0,0 +1,8 @@
+import service from "../utils/request.js";
+
+export const captcha = () => {
+    return service({
+        url: '/authority/getAuthorityList',
+        method: 'post'
+    })
+}

BIN
src/assets/logo.png


+ 1 - 0
src/assets/vue.svg

@@ -0,0 +1 @@
+<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--logos" width="37.07" height="36" preserveAspectRatio="xMidYMid meet" viewBox="0 0 256 198"><path fill="#41B883" d="M204.8 0H256L128 220.8L0 0h97.92L128 51.2L157.44 0h47.36Z"></path><path fill="#41B883" d="m0 0l128 220.8L256 0h-51.2L128 132.48L50.56 0H0Z"></path><path fill="#35495E" d="M50.56 0L128 133.12L204.8 0h-47.36L128 51.2L97.92 0H50.56Z"></path></svg>

+ 76 - 0
src/auto-import.d.ts

@@ -0,0 +1,76 @@
+/* eslint-disable */
+/* prettier-ignore */
+// @ts-nocheck
+// noinspection JSUnusedGlobalSymbols
+// Generated by unplugin-auto-import
+// biome-ignore lint: disable
+export {}
+declare global {
+  const EffectScope: typeof import('vue')['EffectScope']
+  const computed: typeof import('vue')['computed']
+  const createApp: typeof import('vue')['createApp']
+  const customRef: typeof import('vue')['customRef']
+  const defineAsyncComponent: typeof import('vue')['defineAsyncComponent']
+  const defineComponent: typeof import('vue')['defineComponent']
+  const effectScope: typeof import('vue')['effectScope']
+  const getCurrentInstance: typeof import('vue')['getCurrentInstance']
+  const getCurrentScope: typeof import('vue')['getCurrentScope']
+  const h: typeof import('vue')['h']
+  const inject: typeof import('vue')['inject']
+  const isProxy: typeof import('vue')['isProxy']
+  const isReactive: typeof import('vue')['isReactive']
+  const isReadonly: typeof import('vue')['isReadonly']
+  const isRef: typeof import('vue')['isRef']
+  const markRaw: typeof import('vue')['markRaw']
+  const nextTick: typeof import('vue')['nextTick']
+  const onActivated: typeof import('vue')['onActivated']
+  const onBeforeMount: typeof import('vue')['onBeforeMount']
+  const onBeforeRouteLeave: typeof import('vue-router')['onBeforeRouteLeave']
+  const onBeforeRouteUpdate: typeof import('vue-router')['onBeforeRouteUpdate']
+  const onBeforeUnmount: typeof import('vue')['onBeforeUnmount']
+  const onBeforeUpdate: typeof import('vue')['onBeforeUpdate']
+  const onDeactivated: typeof import('vue')['onDeactivated']
+  const onErrorCaptured: typeof import('vue')['onErrorCaptured']
+  const onMounted: typeof import('vue')['onMounted']
+  const onRenderTracked: typeof import('vue')['onRenderTracked']
+  const onRenderTriggered: typeof import('vue')['onRenderTriggered']
+  const onScopeDispose: typeof import('vue')['onScopeDispose']
+  const onServerPrefetch: typeof import('vue')['onServerPrefetch']
+  const onUnmounted: typeof import('vue')['onUnmounted']
+  const onUpdated: typeof import('vue')['onUpdated']
+  const onWatcherCleanup: typeof import('vue')['onWatcherCleanup']
+  const provide: typeof import('vue')['provide']
+  const reactive: typeof import('vue')['reactive']
+  const readonly: typeof import('vue')['readonly']
+  const ref: typeof import('vue')['ref']
+  const resolveComponent: typeof import('vue')['resolveComponent']
+  const shallowReactive: typeof import('vue')['shallowReactive']
+  const shallowReadonly: typeof import('vue')['shallowReadonly']
+  const shallowRef: typeof import('vue')['shallowRef']
+  const toRaw: typeof import('vue')['toRaw']
+  const toRef: typeof import('vue')['toRef']
+  const toRefs: typeof import('vue')['toRefs']
+  const toValue: typeof import('vue')['toValue']
+  const triggerRef: typeof import('vue')['triggerRef']
+  const unref: typeof import('vue')['unref']
+  const useAttrs: typeof import('vue')['useAttrs']
+  const useCssModule: typeof import('vue')['useCssModule']
+  const useCssVars: typeof import('vue')['useCssVars']
+  const useId: typeof import('vue')['useId']
+  const useLink: typeof import('vue-router')['useLink']
+  const useModel: typeof import('vue')['useModel']
+  const useRoute: typeof import('vue-router')['useRoute']
+  const useRouter: typeof import('vue-router')['useRouter']
+  const useSlots: typeof import('vue')['useSlots']
+  const useTemplateRef: typeof import('vue')['useTemplateRef']
+  const watch: typeof import('vue')['watch']
+  const watchEffect: typeof import('vue')['watchEffect']
+  const watchPostEffect: typeof import('vue')['watchPostEffect']
+  const watchSyncEffect: typeof import('vue')['watchSyncEffect']
+}
+// for type re-export
+declare global {
+  // @ts-ignore
+  export type { Component, ComponentPublicInstance, ComputedRef, DirectiveBinding, ExtractDefaultPropTypes, ExtractPropTypes, ExtractPublicPropTypes, InjectionKey, PropType, Ref, MaybeRef, MaybeRefOrGetter, VNode, WritableComputedRef } from 'vue'
+  import('vue')
+}

+ 14 - 0
src/components.d.ts

@@ -0,0 +1,14 @@
+/* eslint-disable */
+// @ts-nocheck
+// Generated by unplugin-vue-components
+// Read more: https://github.com/vuejs/core/pull/3399
+export {}
+
+/* prettier-ignore */
+declare module 'vue' {
+  export interface GlobalComponents {
+    HelloWorld: typeof import('./components/HelloWorld.vue')['default']
+    RouterLink: typeof import('vue-router')['RouterLink']
+    RouterView: typeof import('vue-router')['RouterView']
+  }
+}

+ 44 - 0
src/components/HelloWorld.vue

@@ -0,0 +1,44 @@
+<script setup>
+import { ref } from 'vue'
+
+defineProps({
+  msg: String,
+})
+
+const count = ref(0)
+
+</script>
+
+<template>
+  <h1>{{ msg }}</h1>
+
+  <div class="card">
+    <button type="button" @click="count++">count is {{ count }}</button>
+    <p>
+      Edit
+      <code>components/HelloWorld.vue</code> to test HMR
+    </p>
+  </div>
+
+  <p>
+    Check out
+    <a href="https://vuejs.org/guide/quick-start.html#local" target="_blank"
+      >create-vue</a
+    >, the official Vue + Vite starter
+  </p>
+  <p>
+    Learn more about IDE Support for Vue in the
+    <a
+      href="https://vuejs.org/guide/scaling-up/tooling.html#ide-support"
+      target="_blank"
+      >Vue Docs Scaling up Guide</a
+    >.
+  </p>
+  <p class="read-the-docs">Click on the Vite and Vue logos to learn more</p>
+</template>
+
+<style scoped>
+.read-the-docs {
+  color: #888;
+}
+</style>

+ 16 - 0
src/main.js

@@ -0,0 +1,16 @@
+import { createApp } from 'vue'
+import './style.css'
+import App from './App.vue'
+import router from "@/router/index.js";
+import { createPinia } from 'pinia'
+import ElementPlus from 'element-plus'
+import 'element-plus/dist/index.css'
+import vantage from 'vant'
+import 'vant/lib/index.css';
+const manage = createPinia()
+const app = createApp(App)
+app.use(manage)
+app.use(ElementPlus)
+app.use(router)
+app.use(vantage)
+app.mount('#app')

+ 7 - 0
src/pinia/index.js

@@ -0,0 +1,7 @@
+import { createPinia } from 'pinia'
+
+const store = createPinia()
+
+export {
+    store
+}

+ 28 - 0
src/pinia/modules/user.js

@@ -0,0 +1,28 @@
+import { defineStore } from 'pinia'
+
+export const useUserStore = defineStore('user',() => {
+    const token = ref('')
+    const userInfo = reactive({
+        userId: 0,
+        userName: '',
+        roleId: 0,
+        roleName: ''
+    })
+    const expiresAt = ref(0)
+    const updateInfo = (data) => {
+        token.value = data.token
+        expiresAt.value = data.expiresAt
+        Object.assign(userInfo, {
+            userId: data.user.ID,
+            userName: data.user.nickName,
+            roleId: data.user.authority.authorityId,
+            roleName: data.user.authority.authorityName
+        })
+    }
+    return {
+        token,
+        userInfo,
+        expiresAt,
+        updateInfo
+    }
+})

+ 52 - 0
src/router/index.js

@@ -0,0 +1,52 @@
+import { createRouter, createWebHistory } from "vue-router";
+
+
+// @ts-ignore
+let routes= [
+    {
+        path: '/',
+        name: 'login',
+        //使用import可以路由懒加载,如果不使用,太多组件一起加载会造成白屏
+        component: () => import('@/view/login/login.vue')
+    },
+    {
+        path: '/navigation',
+        name: 'navigation',
+        component: () => import('@/view/navigation/navigation.vue'),
+        children: [
+            {
+                path: 'dashboard',
+                name: 'dashboard',
+                component: () => import('@/view/dashboard/dashboard.vue')
+            },
+            {
+                path: 'project',
+                name: 'project',
+                component: () => import('@/view/project/project.vue')
+            },
+            {
+                path: 'finance',
+                name: 'finance',
+                component: () => import('@/view/finance/finance.vue')
+            },
+            {
+                path: 'approve',
+                name: 'approve',
+                component: () => import('@/view/approve/approve.vue')
+            },
+        ]
+    },
+    //{
+    //配置404页面
+    //path: '/:catchAll(.*)',
+    //name: '404',
+    //component: () => import(''),
+    //}
+]
+// 路由
+const router = createRouter({
+    history: createWebHistory(),
+    routes
+})
+// 导出
+export default router

+ 79 - 0
src/style.css

@@ -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;*/
+/*  }*/
+/*}*/

+ 51 - 0
src/utils/request.js

@@ -0,0 +1,51 @@
+// /utils/request.ts
+
+import axios from 'axios'
+import {useUserStore} from '@/pinia/modules/user'
+const userStore = useUserStore()
+//创建axios实例
+const request = axios.create({
+    baseURL: import.meta.env.VITE_BASE_API,
+    // timeout: 9999999,
+})
+//请求拦截
+request.interceptors.request.use((config) => {
+    const userStore = useUserStore()
+    if (userStore.token !== '') {
+        config.headers.token = userStore.token
+    }
+    return config
+})
+
+//响应拦截
+request.interceptors.response.use(
+    (response) => {
+        return response.data
+    },
+    (error) => {
+        let message = ''
+        const status = error.response.status
+        switch (status) {
+            case 401:
+                message = 'TOKEN过期'
+                break
+            case 403:
+                message = '无权访问'
+                break
+            case 404:
+                message = '请求地址错误'
+                break
+            case 500:
+                message = '服务器出现问题'
+                break
+            default:
+                message = '网络出现问题'
+                break
+        }
+        //提示错误信息
+        //...
+        return Promise.reject(error)
+    },
+)
+//对外暴露
+export default request

+ 13 - 0
src/view/approve/approve.vue

@@ -0,0 +1,13 @@
+<script setup>
+defineOptions({
+  name: 'Approve',
+})
+</script>
+
+<template>
+  <div>审核列表</div>
+</template>
+
+<style scoped>
+
+</style>

+ 13 - 0
src/view/dashboard/dashboard.vue

@@ -0,0 +1,13 @@
+<script setup>
+defineOptions({
+  name: 'Dashboard',
+})
+</script>
+
+<template>
+
+</template>
+
+<style scoped>
+
+</style>

+ 14 - 0
src/view/finance/finance.vue

@@ -0,0 +1,14 @@
+<script setup>
+defineOptions({
+  name: 'finance',
+})
+
+</script>
+
+<template>
+
+</template>
+
+<style scoped>
+
+</style>

+ 114 - 0
src/view/login/login.vue

@@ -0,0 +1,114 @@
+<template>
+  <div>
+    <div class="box">
+      <div class="contentBox">
+        <van-row justify="center">
+          <van-col>
+            <van-image
+                width="6rem"
+                height="6rem"
+                :src="logo"
+            />
+          </van-col>
+        </van-row>
+        <van-form style="width: 90vw;margin: 4vh 0 0 5vw">
+          <van-cell-group inset>
+            <van-field v-model="loginFormData.username" label="用户名" placeholder="请输入用户名" size="large"/>
+            <van-field v-model="unencrypted" label="密码" placeholder="请输入密码" size="large"/>
+            <van-field v-model="loginFormData.captcha" placeholder="请输入验证码" size="large">
+              <template #button>
+                <van-image :src="captchaPicture" width="25vw" height="4vh" @click="getCaptcha"/>
+              </template>
+            </van-field>
+          </van-cell-group>
+        </van-form>
+        <van-row style="margin-top: 4vh">
+          <van-col :span="20" :offset="2">
+            <van-button type="primary" size="large" @click="userLogin">登录</van-button>
+          </van-col>
+        </van-row>
+      </div>
+    </div>
+  </div>
+</template>
+
+<script setup>
+import { captcha, routerPublicKey, register } from '@/api/login.js'
+import logo from '@/assets/logo.png'
+import { JSEncrypt } from 'jsencrypt'
+import { useUserStore } from '@/pinia/modules/user.js'
+const router = useRouter()
+
+const userStore = useUserStore()
+
+defineOptions({
+  name: 'Login',
+})
+onMounted(() => {
+  getCaptcha()
+  getPublicKey()
+})
+// 未加密密码
+const unencrypted = ref('')
+// 登录数据
+const loginFormData = reactive({
+  username: '',
+  password: '',
+  captcha: '',
+  captchaId: ''
+})
+// 验证码图片
+const captchaPicture = ref('')
+// 获取验证码
+const getCaptcha = () => {
+  captcha().then(res => {
+    if (res.code === 0) {
+      captchaPicture.value = res.data.picPath
+      loginFormData.captchaId = res.data.captchaId
+    }
+  })
+}
+
+// 公钥
+const publicKey = ref('')
+// 获取公钥
+const getPublicKey = () => {
+  routerPublicKey().then(res => {
+    if (res.code === 0) {
+      publicKey.value = res.data.publicKey
+    }
+  })
+}
+
+// 登录
+const userLogin = () => {
+  // 密码rsa加密
+  let encryptor = new JSEncrypt()
+  encryptor.setPublicKey(publicKey.value)
+  loginFormData.password = encryptor.encrypt(unencrypted.value)
+  // 执行登录
+  register(loginFormData).then(res => {
+    if (res.code === 0) {
+      const data = res.data
+      userStore.updateInfo(res.data)
+      router.push({path: '/navigation'})
+    }
+  })
+}
+
+</script>
+
+<style scoped lang="scss">
+.box {
+  width: 100vw;
+  height: 100vh;
+  background-color: #f7f8fa;
+  display: flex;
+  align-items: center;
+  justify-content: center;
+  .contentBox{
+    width: 100vw;
+    height: 50vh;
+  }
+}
+</style>

+ 72 - 0
src/view/navigation/navigation.vue

@@ -0,0 +1,72 @@
+<script setup>
+import {useUserStore} from '@/pinia/modules/user.js'
+import {asyncMenu} from '@/api/menu.js'
+const router = useRouter()
+const userStore = useUserStore()
+onMounted(() => {
+  getMenuList()
+})
+// 导航页标题
+const pageTitle = ref('')
+// 导航弹窗显示
+const navigatorShow = ref(false)
+const openNavigation = () => {
+  navigatorShow.value = true
+}
+const navigationList = reactive([])
+const getMenuList = () => {
+  asyncMenu().then(res => {
+    if (res.code === 0) {
+      const list = res.data.menus
+      list.forEach(item => {
+        if ([1, 41, 45, 55].indexOf(item.ID) > -1) {
+          let option = {
+            value: item.ID,
+            text: item.meta.title
+          }
+          navigationList.push(option)
+        }
+      })
+    }
+  })
+}
+
+const navigatorJump = (val) => {
+  const mine = val.selectedOptions[0].value
+  let obj = {
+    1: {address: '/navigation/dashboard', title:'仪表盘'},
+    41: {address: '/navigation/project', title:'项目文件管理'},
+    45: {address: '/navigation/finance', title:'费用管理'},
+    55: {address: '/navigation/approve', title:'审核管理'},
+  }
+  router.push({path: obj[mine].address})
+  pageTitle.value = obj[mine].title
+  navigatorShow.value = false
+}
+</script>
+
+<template>
+  <van-nav-bar left-text="返回" left-arrow>
+    <template #title>
+      <span style="font-weight: 500">{{pageTitle}}</span>
+    </template>
+    <template #right>
+      <van-icon name="bars" size="2rem" @click="openNavigation"/>
+    </template>
+  </van-nav-bar>
+  <router-view/>
+  <van-popup v-model:show="navigatorShow" round position="bottom">
+    <van-picker
+        :columns="navigationList"
+        @cancel="navigatorShow = false"
+        @confirm="navigatorJump"
+    />
+  </van-popup>
+</template>
+
+<style scoped lang="scss">
+.topNavigation {
+  height: 7vh;
+  border-bottom: 1px solid #000000;
+}
+</style>

+ 247 - 0
src/view/project/project.vue

@@ -0,0 +1,247 @@
+<script setup>
+import { getAllProject, getProjectFile, getFileType, createProjectFile } from '@/api/project.js'
+import { Download, Upload } from '@element-plus/icons-vue'
+import { showNotify } from "vant";
+defineOptions({
+  name: 'Project',
+})
+onMounted(() => {
+  queryProjectList()
+  queryFileType()
+})
+const projectList = reactive([])
+const projectValue = ref(0)
+const queryProjectList = () => {
+  getAllProject().then(res => {
+    if (res.code === 0) {
+      const list = res.data
+      projectList.length = 0
+      projectList.push(...list)
+      queryProjectFile(list[0].code)
+    }
+  })
+}
+// 项目文件
+const fileCondition = reactive({
+      code: "",
+      name: "",
+      genre: 0,
+      pageInfo: {
+        page: 1,
+        pageSize: 10
+      }
+    }
+)
+const fileList = reactive([])
+const queryProjectFile = (code) => {
+  fileCondition.code = code
+  getProjectFile(fileCondition).then(res => {
+    if (res.code === 0) {
+      const list = res.data.list
+      fileList.length = 0
+      fileList.push(...list)
+    }
+  })
+}
+const finished = ref(false)
+const refreshList = () => {}
+
+const selectAll = ref(false)
+
+//上传文件
+const uploadFileShow = ref(false)
+const fileTypeList = reactive([])
+const fileTypeText = ref('')
+const fileTypeValue = ref(0)
+const fileTypeShow = ref(false)
+const fileObjectList = ref([])
+const accept = ref('.doc,.docx,.pdf,.ppt,.pptx,.xlsx,.xls,image/*')
+const uploadFile = () => {
+  if (fileTypeText.value.length === 0) {
+    showNotify({ type: 'danger', message: '请选择文件类型' });
+  } else if (fileObjectList.value.length === 0) {
+    showNotify({ type: 'danger', message: '暂未选择文件' });
+  }
+  // const formData = new FormData()
+  // fileObjectList.value.forEach(item => {
+  //   formData.append('file', item)
+  // })
+  // formData.append('genre', fileTypeValue.value)
+  // formData.append('code', fileCondition.code)
+  // createProjectFile(formData).then(res => {
+  //   if (res.code === 0) {
+  //     console.log(res.data)
+  //   }
+  // })
+}
+const customFieldName = {
+  text: 'name',
+  value: 'ID',
+}
+// 获取文件类型
+const queryFileType = () => {
+  getFileType().then(res => {
+    if (res.code === 0) {
+      const list = res.data
+      fileTypeList.length = 0
+      fileTypeList.push(...list)
+      // let option = { text: '', value: '' }
+      // list.forEach(item => {
+      //   option.text = item.name
+      //   option.value = item.ID
+      //   fileTypeList.push(option)
+      // })
+      // console.log('数据', fileTypeList)
+    }
+  })
+}
+
+const confirmFileType = ({ selectedValues }) => {
+  fileTypeValue.value = selectedValues[0]
+  fileTypeList.forEach(item => {
+    if (item.ID === selectedValues[0]) {
+      fileTypeText.value = item.name
+    }
+  })
+  fileTypeShow.value = false
+}
+</script>
+
+<template>
+  <van-row style="margin-top: 2vh">
+    <van-col :span="14" :offset="1">
+      <el-select placeholder="请选择项目" size="large">
+        <el-option
+            v-for="item in projectList"
+            :key="item.code"
+            :label="item.name"
+            :value="item.code"
+        >
+        </el-option>
+      </el-select>
+    </van-col>
+    <van-col :span="7" :offset="1">
+      <el-dropdown split-button type="primary" size="large">
+        操作
+        <template #dropdown>
+          <el-dropdown-menu>
+            <el-dropdown-item>
+              <el-button type="primary" :icon="Upload" @click="uploadFileShow = true">上传</el-button>
+            </el-dropdown-item>
+            <el-dropdown-item>
+              <el-button type="primary" :icon="Download">下载</el-button>
+            </el-dropdown-item>
+          </el-dropdown-menu>
+        </template>
+      </el-dropdown>
+    </van-col>
+  </van-row>
+  <van-row style="margin-top: 3vh">
+    <van-col :span="11" :offset="1">
+      <van-button icon="filter-o" size="small">文件筛选</van-button>
+    </van-col>
+    <van-col :span="11" class="design">
+      <van-checkbox v-model="selectAll" shape="square" label-position="left">全选</van-checkbox>
+    </van-col>
+  </van-row>
+  <van-row style="margin-top: 2vh">
+    <van-col :span="24">
+      <van-list
+          finished-text="没有更多了"
+          @load="refreshList"
+          :finished="finished"
+      >
+        <van-cell-group>
+          <van-cell v-for="item in fileList" :key="item.ID" size="large">
+            <template #title>
+              <el-text size="large">{{item.name}}</el-text>
+            </template>
+            <template #value>
+              <van-checkbox shape="square" style="margin-left: 41vw"></van-checkbox>
+            </template>
+          </van-cell>
+        </van-cell-group>
+      </van-list>
+    </van-col>
+  </van-row>
+  <van-popup
+      v-model:show="uploadFileShow"
+      style="width: 87vw;height: 58vh"
+      round
+      closeable
+      close-icon-position="top-right"
+  >
+    <van-row style="margin-top: 2vh">
+      <van-col :span="16" :offset="1">
+        <span style="font-size: 1.1rem;">上传文件</span>
+      </van-col>
+    </van-row>
+    <van-row style="margin-top: 3vh">
+      <van-col :span="8" :offset="1">
+        <van-button
+            type="primary"
+            plain
+            icon="arrow-down"
+            @click="fileTypeShow = true">
+          文件类型
+        </van-button>
+      </van-col>
+      <van-col :span="13" :offset="1" class="typeLayout">
+        <van-field
+            v-model="fileTypeText"
+            disabled
+            placeholder="请选择文件类型"
+            style="border: 1px solid #dadadd;border-radius: 5px"/>
+<!--        <el-text :line-clamp="1" style="font-size: 1rem">-->
+<!--          {{fileTypeText}}-->
+<!--        </el-text>-->
+      </van-col>
+    </van-row>
+    <van-row>
+      <van-col :span="22" :offset="1">
+        <div class="fileExhibition">
+          <van-uploader
+              v-model="fileObjectList"
+              multiple
+              :accept="accept"
+          />
+        </div>
+      </van-col>
+    </van-row>
+    <van-row style="margin-top: 2vh">
+      <van-col :span="22" :offset="1" class="design">
+        <van-button type="primary" @click="uploadFile">确认</van-button>
+      </van-col>
+    </van-row>
+  </van-popup>
+  <van-popup position="bottom" v-model:show="fileTypeShow" round>
+    <van-picker
+        :columns="fileTypeList"
+        :columns-field-names="customFieldName"
+        @cancel="fileTypeShow = false"
+        @confirm="confirmFileType"
+    />
+  </van-popup>
+</template>
+
+<style scoped lang="scss">
+.design{
+  display: flex;
+  align-items: center;
+  justify-content: end;
+}
+.typeLayout{
+  display: flex;
+  align-items: center;
+  justify-content: center;
+}
+
+.fileExhibition{
+  margin-top: 3vh;
+  height: 33vh;
+  border: 1px dotted #c8c9cc;
+  Overflow: auto;
+  box-sizing: border-box;
+  padding: 3vw;
+}
+</style>

+ 58 - 0
vite.config.js

@@ -0,0 +1,58 @@
+import {defineConfig} from 'vite'
+import vue from '@vitejs/plugin-vue'
+import * as path from 'path'
+// 自动导入vue中hook reactive ref等
+import AutoImport from "unplugin-auto-import/vite"
+//自动导入ui-组件 比如说ant-design-vue  element-plus等
+import Components from 'unplugin-vue-components/vite'
+
+// https://vitejs.dev/config/
+export default defineConfig({
+    plugins: [
+        vue(),
+        AutoImport({
+            //安装两行后你会发现在组件中不用再导入ref,reactive等
+            imports: ['vue', 'vue-router'],
+            //存放的位置
+            dts: "src/auto-import.d.ts",
+        }),
+        Components({
+            // 引入组件的,包括自定义组件
+            // 存放的位置
+            dts: "src/components.d.ts",
+        }),
+    ],
+    resolve: {
+        alias: {
+            '@': path.resolve(__dirname, './src'),
+        },
+    },
+    // 解决跨域问题
+    server: {
+        host: '0.0.0.0',
+        // port: 4000,
+        proxy: {
+            '/api': {
+                target: 'http://192.168.110.218:8220',
+                // secure: false, // 请求是否为https
+                changeOrigin: true,
+                rewrite: (path) => path.replace(/^\/api/, '') //api替换为'',
+            },
+        },
+    },
+
+
+    // resolve: {
+    //   alias: {
+    //     '@': path.resolve(__dirname, 'src'),
+    //   },
+    // },
+    css: {
+        preprocessorOptions: {
+            less: {
+                additionalData: `@import "@/assets/style/base.less";`,
+            }
+        },
+    },
+})
+