不依赖框架用vue3空白项目从头打造一个过得去的前端

Posted 左直拳

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了不依赖框架用vue3空白项目从头打造一个过得去的前端相关的知识,希望对你有一定的参考价值。

通过全程参与,可以加深对VUE项目的理解。

近期做的一个项目,前端除了UI外,没有使用什么框架。不使用现成的框架是无奈之举,因为找不到合适的。之前用的框架,比较老旧,还是vue2的;新的吧,有学习成本,怕耽误时间,也不知道效果怎么样,存在一定的风险。利用最基本的“空白”项目,按需添加基础功能,代码可控,进度也较有保证,同时还能够消除无框架不会工作恐惧症。现在记录一下心得,以后可以反复使用。

记录重点有:

0、整体结构
1、路由
2、导航条及子菜单
3、ajax请求封装
4、vue.config.js及系统配置
5、登录及退出

一、整体结构

1、创建项目
首先是新创建一个vue3项目。方法是

vue create 项目名称

然后选择合适的选项。具体可参考拙作
vue3多个项目共享开发和单个项目独立打包的解决方案

1)按默认方案创建

项目结构:

2)创建时增加路由及store支持


多了router、store以及一些页面。

3)我们项目的整体结构
实际项目中,当然还会夹带一些私货,额外增加一些东东。

2、项目入口main.js
注意项目的入口不是App.vue,而是main.js。

import  createApp  from 'vue'
import App from './App.vue'
import router from './router'
import store from './store'

createApp(App).use(store).use(router).mount('#app')

main.js是系统约定好的名称,正如一般程序的入口是函数void main()一样。整个项目的入口是main.js,然后每个模块的入口可以是index.js。都是js。本质上,vue是一个大的js语法糖。它有这样那样的结构,让人只把杭州作汴州,但归根到底,它最终是要编译成原始的js,才能被浏览器识别、运行。

从上述代码可以知道,main.js的作用是加载App.vue,引入store、路由,然后绑定到页面id="app"的div。这样该div就是整个项目的活动区域,即展示页面内容的区域了:

vue项目是单页面应用,只有一张html页面。我们看到的所有内容,都展示在id="app"的这个div上!完全由js控制。

3、我们项目里的main.js
main.js这里的引用,都是全局性的。除了路由,store,还可以引入ui框架,css,全局性组件等等。比如我们项目里的main.js是这样写:

import  createApp  from "vue";
import App from "./App.vue";
import router from "./router";

import PerfectScrollbar from "vue3-perfect-scrollbar";//滚动条美化
import "vue3-perfect-scrollbar/dist/vue3-perfect-scrollbar.css";
import Antd from "ant-design-vue";//ant design vue,UI框架
import "ant-design-vue/dist/antd.css";

import "@/assets/css/default.css"; //自定义的全局css

createApp(App).use(Antd).use(PerfectScrollbar).use(router).mount("#app");

二、路由

简单来说,路由就是菜单配置,将菜单项的id,路径,都集中写在了一个配置文件里。

1、系统自动生成的路由

import  createRouter, createWebHashHistory  from 'vue-router'
import HomeView from '../views/HomeView.vue'

const routes = [
  
    path: '/',
    name: 'home',
    component: HomeView
  ,
  
    path: '/about',
    name: 'about',
    // route level code-splitting
    // this generates a separate chunk (about.[hash].js) for this route
    // which is lazy-loaded when the route is visited.
    component: () => import(/* webpackChunkName: "about" */ '../views/AboutView.vue')
  
]

const router = createRouter(
  history: createWebHashHistory(),
  routes
)

export default router

以上完全是系统自动生成的代码。看上去也不难理解。其中路由表routes对应的是导航条

2、实际项目中的路由
见本文第六章第一条

3、代码中使用路由
如果想多加几个导航条,可以依葫芦画瓢,很容易就能实现。每个路由,name是唯一的ID,在代码里控制跳转的话,引用name就可以,可以不再重复写这个path。比如在某个vue里,可以这样使用路由:

import  defineComponent  from "vue";
import  useRouter  from "vue-router"; //引入useRouter

export default defineComponent(
  setup() 
	const router = useRouter();
    const browseIt = (fd) => 
      const to = router.resolve(
        name: "about", //这里是跳转页面的name,要与路由设置保持一致
        params:  id: fd.id ,
      );
      window.open(to.href, "_blank");//新开一个页面,打开about
    ;

    return 
      browseIt,
    ;
  ,
);

三、导航条及子菜单

1、导航条
系统生成的代码,已经做了很好的示范:

<template>
  <!-- 导航条 -->
  <nav>
    <router-link to="/">Home</router-link> |
    <router-link to="/about">About</router-link>
  </nav>
  
  <!-- 导航的目的地。注意该部分必不可少 -->
  <router-view/>
</template>

通常,我们的菜单可以从服务器端返回,然后利用循环语句输出。

2、子菜单
导航条是一级菜单,二级或以下是子菜单。子菜单,可以利用UI框架来完成。比如我们利用ant design vue的menu组件来完成

四、ajax请求封装

我们做的项目,总免不了要从服务器端请求数据。前后端分离,理论上前端的数据都来自于服务器端。

目前一般是结合第三方组件axios对ajax请求进行封装。理由主要有2个:

1)ajax一般有超时、返回代码区别对待等共性操作,封装一个ajax处理方法,统一调用,方便维护和修改;除此之外,axios还可以对ajax请求进行拦截,使得请求前、请求后、结果返回做相应处理。

2)解决跨域问题。axios本身似乎并不解决跨域问题,但由于第一点,它对ajax请求进行了封装,我们可以利用一个统一方法,结合api路径前缀做转发,使得浏览器以为api所在路径与前端是同一台服务器,因而不存在跨域。注意所谓跨域问题,是浏览器的一个安全设置,是一种保护措施。只要它不觉得跨域,那跨域就不存在。

1、统一的ajax封装方法

1)ajax封装代码(src/request/index.js)

import axios from "axios";

// 创建一个 axios 实例
const service = axios.create(
  baseURL: "/api", // 所有的请求地址前缀部分
  timeout: 60000, // 请求超时时间毫秒
  withCredentials: true, // 异步请求携带cookie
  headers: 
    // 设置后端需要的传参类型
    "Content-Type": "application/json",
    //'token': 'your token',
    "X-Requested-With": "XMLHttpRequest",
  ,
);

// 添加请求拦截器
service.interceptors.request.use(
  function (config) 
    // 在发送请求之前做些什么
    return config;
  ,
  function (error) 
    // 对请求错误做些什么
    console.log(error);
    return Promise.reject(error);
  
);

// 添加响应拦截器
service.interceptors.response.use(
  function (response) 
    console.log(response);
    // 2xx 范围内的状态码都会触发该函数。
    // 对响应数据做点什么
    // dataAxios 是 axios 返回数据中的 data
    const dataAxios = response.data;
    // 这个状态码是和后端约定的
    //const code = dataAxios.reset;
    return dataAxios;
  ,
  function (error) 
    // 超出 2xx 范围的状态码都会触发该函数。
    // 对响应错误做点什么
    console.log(error);
    return Promise.reject(error);
  
);

export default service;

2)使用ajax统一封装方法

import request from "@/request";

export const userLogin = (params) => 
  return request( //request就是统一封装好的方法
    url: "/sys/login",//注意request方法会在前面加上前缀“/api”,变成实质上是请求 “/api/sys/login”
    params,
    method: 'post'
  )

2、开发环境中,利用vue.config.js做api路径转发
如上所属,ajax的封装方法request,会在api请求路径前加上前缀“/api”,可以利用这一点来设置转发。转发是为了避免跨域。其原理,表面上,我们请求的是当前服务器的路径“/api/a/b/c”,但我们设置凡“/api”开头的路径,都转发到另一台服务器(即后端所在服务器)。浏览器蒙在鼓里,并没有察觉,因此不会触发所谓跨域警告。

  devServer: 
    //devServer 只是一个webpack插件 只能用于开发环境
    proxy: 
      "/api": 
        target: "192.168.0.22",
        pathRewrite: 
          "^/api": "",
        ,
      ,
    ,
  ,

3、生产环境中,利用nginx做api路径转发

location /api/ 
	proxy_pass http://192.168.0.22:8090/;#必须斜杠/结尾
	proxy_set_header   X-Forwarded-Proto $scheme;
	proxy_set_header   Host              $http_host;
	proxy_set_header   X-Real-IP         $remote_addr;

五、vue.config.js及系统配置

有关vue.config.js里面的配置,上面约略提到了一些,这里给出完整代码:

1、系统生成的vue.config.js代码

const  defineConfig  = require('@vue/cli-service')
module.exports = defineConfig(
  transpileDependencies: true
)

2、按项目需要修改后的代码

const  defineConfig  = require("@vue/cli-service");
const path = require("path");
const appConfig = require("./public/config");

const resolve = (dir) => 
  return path.join(__dirname, dir);
;

module.exports = defineConfig(
  transpileDependencies: true,

  devServer: 
    //devServer 只是一个webpack插件 只能用于开发环境
    proxy: 
      "/api": 
        target: appConfig.server,
        pathRewrite: 
          "^/api": "",
        ,
      ,
    ,
  ,

  // 项目部署基础
  // 默认情况下,我们假设你的应用将被部署在域的根目录下,
  // 例如:https://www.my-app.com/
  // 默认:'/'
  // 如果您的应用程序部署在子路径中,则需要在这指定子路径
  // 例如:https://www.foobar.com/my-app/
  // 需要将它改为'/my-app/'
  publicPath: "/",
  chainWebpack: (config) => 
    config.resolve.alias
      .set("@", resolve("src")) // key,value自行定义,比如.set('@@', resolve('src/components'))
      .set("_c", resolve("src/components"));

    config.plugin("html").tap((args) => 
      args[0].title = appConfig.app.name;
      return args;
    );
  ,
);

值得一提的是,里面的devServer元素设置,针对的是开发环境。详见拙作:vue.config.js中的devServer

3、真正的项目配置
按我的理解,vue.config.js只在开发阶段和发布时有用,之后就像被消费了的耗材,没啥用处了。同一项目中,真正的项目配置,是/public/config/index.js,即使是发布、部署到生产环境,仍然可以修改,是真正意义上的配置文件。

/public/config/index.js

exports.app = 
  name: "订餐拿饭抓阄系统",
  owner: "蓬蓬养猪场",
  developer: "一群饭桶",
;

详见拙作:vue项目读取全局配置

六、登录及退出

主要是路由的应用。原理是:
将页面分为无须登录可浏览和必须登录方可浏览2种。在路由表中做过滤。当转向必须登录页面时,检查登录状态,如果已登录,放行;未登录,转向登录页。注意登录页要设为无须登录可浏览。流程很简单,大家都明白,就不画流程图了。

1、路由表(/src/router/index.js)
完整的路由表。有关登录控制部分,见所谓“路由守卫”。

import  createRouter, createWebHashHistory  from "vue-router";
import Home from "../views/home/PageIndex.vue";

const routes = [
  
    path: "/login",
    name: "login",
    component: () => import("../views/login/PageIndex.vue"),
    meta: 
      noLogin: true, //无须登录即可浏览。自定义属性
    ,
  ,
  
    path: "/",
    name: "Home",
    component: Home,
  ,
  
    path: "/map",
    name: "Map",
    component: () => import("../views/map/PageIndex.vue"),
  ,
  
    path: "/resource",
    name: "Resource",
    component: () => import("../views/resource/PageIndex.vue"),
  ,
  
    path: "/resource/detail/:id",
    name: "ResourceDetail",
    component: () => import("../views/resource/PageDetail.vue"),
  ,
  
    path: "/sys",
    name: "Sys",
    component: () => import("../views/sys/PageIndex.vue"),
  ,
];

const router = createRouter(
  history: createWebHashHistory(),
  routes,
);

// 路由守卫
router.beforeEach((to, from, next) => 
  const isLogin = localStorage.isLogin ? true : false;
  if (isLogin) 
    //已经登录的情况下,不能再打开登录页
    to.path === "/login" ? next("home") : next();
   else 
    //如果无须登录则直接打开,否则转向登录页面
    to.meta.noLogin || to.path === "/login" ? next() : next("/login");
  
);

export default router;

2、登录及登出
注意要使用router本身的方法,不可用原始的js,如window.location.href = *** 这种方法,否则部署到nginx会报错。
1)登录

const router = useRouter();
localStorage.setItem("isLogin", true);
document.cookie = "token=" + res.token;
router.replace( path: "/" );//转向首页。使用replace,避免登录后回退问题

2)登出

const router = useRouter();
localStorage.removeItem("isLogin");
router.replace( path: "/login" );

七、store

暂无

以上是关于不依赖框架用vue3空白项目从头打造一个过得去的前端的主要内容,如果未能解决你的问题,请参考以下文章

Vue3.0 项目启动(打造企业级音乐App)

Vue3.0实现拖拽布局

使用Vite和TypeScript带你从零打造一个属于自己的Vue3组件库

vite+vue3+ts实战项目,教你实现一个网页版的typora!(前端篇)

vue3微信公众号商城项目实战系列开发环境准备

day01-项目介绍+SSM环境搭建