/vue_notes

Learning notes to share, with the version update.

Primary LanguageJavaScriptMIT LicenseMIT

VUE前端框架

背景知识

目前vue2已经不再被维护,但vue3依然继承了很多vue2的理念和接口,我们依然需要先从vue2开始学习。

vue2文档https://v2.vuejs.org/

vue3文档https://vuejs.org/

官方全局状态管理

vuex:https://vuex.vuejs.org/zh/

★pinia:https://pinia.vuejs.org/zh/

事件总线: 如果你还想念事件总线的话。。。

tiny-emitter: https://www.npmjs.com/package/tiny-emitter

mitt: https://www.npmjs.com/package/mitt

官方路由:

★Vue-Router:https://router.vuejs.org/zh/

TanStack-Router:https://tanstack.com/router/v1/docs/overview

官方脚手架

★Vite:https://cn.vitejs.dev/

vue-cli:https://cli.vuejs.org/zh/

vscode插件

vue3:https://cn.vuejs.org/guide/typescript/overview.html#ide-support

vue2:自己找一下吧

浏览器vue插件

https://chrome.google.com/webstore/detail/vuejs-devtools/nhdogjmejiglipccpnnnanhbledajbpd

SFC插件

GraphQL插件1:https://gridsome.org/docs/querying-data/

GraphQL插件2: https://github.com/wheatjs/vite-plugin-vue-gql

高级自定义插件:https://cn.vuejs.org/guide/scaling-up/tooling.html#sfc-custom-block-integrations

jsx语法支持:

跟脚手架配置有关。

vite:https://github.com/vitejs/vite-plugin-vue/tree/main/packages/plugin-vue-jsx

vue-cli:https://github.com/vuejs/babel-plugin-jsx/blob/dev/packages/babel-plugin-jsx/README-zh_CN.md

测试框架:

vite:https://vitest.dev/

Cypress:https://www.cypress.io/

后端框架

Nuxt: https://nuxt.com/

Vite-SSR: vite工具自带插件https://cn.vitejs.dev/guide/ssr.html

好用的拖拽: VueDraggablePlus

vue的特点

什么是渐进式? 就是一开始不需要你完全掌握它的全部功能特性,可以后续逐步增加功能。没有多做职责之外的事情。现在react也是如此。

根据SFC插件+脚手架插件的各种支持,我们应当选择合适的lang=""

SFC组件化模式:

~/components/child.vue

<template lang="pug">
  <!-- 组件模板HTML区域,这里我们开启pug模板渲染模式 -->
  <div class="example">{{ msg }}</div>
</template>

<script lang="ts">
// 组件js区域,这里我们使用ts代码
export default {
  data() {
    return {
      msg: 'Hello world!'
    }
  }
}
</script>

<style lang="less">
/* 样式组件,这里我们使用less样式 */
.example {
  color: red;
}
</style>

<custom1>
  自定义模块,这里不常用,一般用来装载vue专属插件,实现更多功能。
</custom1>

小技巧

# vue3比vue2更好的写法。
<script lang="ts" setup>
import { ref } from 'vue'

// 响应式变量
const count = ref(0)

// 变量
const msg = 'Hello!'

// 函数
function log() {
  console.log(msg)
}
</script>
# scoped标签让样式只影响当前组件
<style lang="less" scoped>
.example {
  color: red;
}
/* 如果想要某个样式影响子节点 */
.a :deep(.b) {
  /* 使用:deep(el) */
}
/* 如果想要某个样式影响插槽 */
:slotted(div) {
  color: red;
}
/* 如果想要某个样式影响全局 */
:global(.red) {
  color: red;
}
</style>

混合全局样式和局部样式

<style>
/* 全局样式 */
</style>

<style scoped>
/* 局部样式 */
</style>

动态组件

<script setup>
import Foo from './Foo.vue'
import Bar from './Bar.vue'
</script>

<template>
  <component :is="Foo" />
  <component :is="someCondition ? Foo : Bar" />
</template>

递归组件

import { FooBar as FooBarChild } from './components'

命名空间组件

<script setup>
import * as Form from './form-components'
</script>

<template>
  <Form.Input>
    <Form.Label>label</Form.Label>
  </Form.Input>
</template>

自定义指令

<script setup>
const vMyDirective = {
  beforeMount: (el) => {
    // 在元素上做些操作
  }
}
</script>
<template>
  <h1 v-my-directive>This is a Heading</h1>
</template>
<script setup>
import { myDirective as vMyDirective } from './MyDirective.js'
</script>

推荐使用ts语法

<script setup>
const props = defineProps({
  foo: String
})

const emit = defineEmits(['change', 'delete'])
// setup 代码
</script>

image-20220625121810770

声明式编码,有点模板化的感觉。

image-20220625121933607

image-20220625122645442

css modules 模块化

推荐使用文件格式style.module.css

但在**.vue**文件中,一个 <style module> 标签会被编译为 CSS Modules 并且将生成的 CSS class 作为 $style 对象暴露给组件:

<template>
  <p :class="$style.red">This should be red</p>
</template>

<style module>
.red {
  color: red;
}
</style>
<template>
  <p :class="classes.red">red</p>
</template>

<style module="classes">
/* 可以带入命名空间 */
.red {
  color: red;
}
</style>

如果想要多次调用模块化css,除了import外,还可以通过一个钩子。

import { useCssModule } from 'vue'

// 在 setup() 作用域中...
// 默认情况下, 返回 <style module> 的 class
useCssModule()

// 具名情况下, 返回 <style module="classes"> 的 class
useCssModule('classes')

同时也支持,动态组件的样式绑定法

<script setup>
const theme = {
  color: 'red'
}
</script>

<template>
  <p>hello</p>
</template>

<style scoped>
p {
  color: v-bind('theme.color');
}
</style>

v2开始学习

避免开发环境提示,我们可以全局配置,

Vue.config.productionTip = false;

特点记忆:

1.Vue实例和容器,一一对应,不存在多对多和一对多。

2.{{xxx}}有种模板语法的感觉,可以写js对象。

3.data中的数据一旦改变,则渲染结果也会改变

image-20220627202143736

image-20220627201727500

插值语法

image-20220627203719798

指令语法

image-20220627205809948

image-20220627210009771

单项绑定

image-20220627214929608

双向绑定

image-20220627215413204

image-20220627215833839

Vue实例化

image-20220627221320385

el和data扩展写法

image-20220627222434161

MVVM模型

image-20220627224614851

本质:数据和模板合并,加入监听和渲染。

最后所有的属性都出现在了VM上

image-20220627225309513

数据代理

image-20220627234953902

事件

image-20220628003844724

修饰符可以连写。

image-20220628105232702

键盘事件

可以连写,组合键

image-20220628130745441

计算属性

image-20220628154103836

image-20220628154442871

事件简写

image-20220628162115941

监视属性

image-20220628164652782

image-20220628173412598

监视,计算的区别?

计算属性可以创建一个实例,然后调用。

监视属性则必须先创建一个data实例属性,然后调用。

监视属性可以放入异步方法,必须可以放入

比如放入一个setTimeout延时器,计算属性需要return,而监视属性直接赋值

image-20220628175052829

this始终指向调用者,当多个函数嵌套时,则默认指向外面一层。

箭头函数和function()的区别,取this的区别。

function()中的this默认指向,外层对象,如果没有则默认window。
箭头函数中的this始终指向最外层对象window。

class和style

image-20220628193521731

条件渲染

image-20220628212138141

列表渲染

image-20220628215548840

虚拟DOM对比算法

image-20220628221110273

image-20220628221822956

image-20220628233614490

Vue监测的底层逻辑

image-20220629001214798

image-20220629002002148

vue中监听的只有属性值,当不通过属性值进行传参的改变时,就会导致渲染失败.

set功能

最主要时可以后期往data添加元素。

image-20220629010302506 image-20220629011611670

表单

image-20220629021326430

image-20220629021433619

过滤器

image-20220629025409759

image-20220629025422745

常用指令

v-html与v-text

image-20220629182126192

image-20220629181025122

v-cloak

image-20220629191020383

v-once

image-20220629191032993

v-pre

image-20220629191639117

自定义指令

image-20220629205012124

自定义指令坑

image-20220629205900731

生命周期

Vue 实例生命周期

组件化

image-20220630002308121

image-20220630002400836

非单文件组件

image-20220630133820765

image-20220630133540891

单文件组件

vue文件封装组件

image-20220630151128020

app管理子组件

image-20220630151105704

各个子组件

image-20220630151153487

image-20220630151141677

但是缺少了脚手架 or webpack

image-20220630151229438

脚手架vue-cli

image-20220630143845837

这里我们使用CLI脚手架

vue指令
# 查看脚手架版本
vue -V
# 创建脚手架,名字回避npm包里的名字
vue create 

命令行会要求你选择选项

vue版本

image-20220630153305396

选择默认包管理

image-20220630155255999

一般选择npm
当然也可以选择yarn

编译文件,并运行一个实例服务器

# 可以多次重启
npm run serve

image-20220630154610142

# 创建一个生产版本的dist
npm run build
# 执行检查文件(才不用)
npm run lint

image-20220630160923065

记录一个错误

image-20220630163209089

再次重启编译服务时,出现以上错误。

重要原因是,脚手架内的lint插件检查到了不符合命名规范的组件

image-20220630163352813

我们需要配置vue.config.js让错误检查关闭

image-20220630163424064

Vue如果不整这么多版本,导致体积太大,所以有

vue.js 完全体js

vue.min.js 压缩过的js

runtime.js 运行时js

轻量级脚手架 Vite

优势

vuex使用

store是vue的管理中心。

image-20220702170953461

一个在VM之外的数据和方法的存放方式,

使用devTool工具查看vuex触发过程

image-20220703002359790

功能看代码。

vue-router路由

单页面,适合展示,后台管理,数据可视化。局部刷新加路由url变化

image-20220703143631481

多页面,适合大型商品网站,资料网站等等。其实也是多个单页面组成 - -

image-20220703183600696

Pinia的使用

Vue 的官方状态管理库已更改为Pinia。Pinia 具有与 Vuex 5 几乎完全相同或增强的 API,在Vuex 5 RFC中进行了描述。您可以简单地将 Pinia 视为具有不同名称的 Vuex 5。Pinia 也适用于 Vue 2.x。

部署

UI组件库

image-20220704095655169

v3

vite新脚手架,vue-cli

这里我们推荐,使用vite构建vue3+Ts开发模块。vite是比webpack更先进的动态脚手架。

$ npm create vite@latest my-vue-ts --template vue-ts

image-20220828220006089

Pinia.js新状态管理工具,vuex

这个构建起来更快。

npm install pinia

https://pinia.vuejs.org/getting-started.html

vue Loader 新格式文件

服务端渲染Server-Side Rendering

1.反向代理服务器。

https://zxuqian.cn/vite-proxy-config/

2.SSR直接代理。用的太少了,一般都是借助nginx对static文件进行代理。

组合式API ,setup

全新的数据和事务的包装形式setup,vue2和vue3语法之间存在较大差异,不要混用。

image-20220704191635793

响应式的不同

image-20220704191521258

操作对象中的属性

需要

Vue.$delete

Vue.$set(obj, key, value)

操作数组中的元素。

则需要数组那七个数组方法,才能触发响应式。

image-20220629011611670

Vue3则不同

image-20220705114143597

vue3生命周期

实例的生命周期

Vue 3 “Vapor Mode”

Vapor 模式是受 Solid.js 启发的一种新的替代编译策略。它旨在通过将代码编译为更高效的 JavaScript 输出来增强应用的性能。

当在应用级别使用时,可以完全删除虚拟 DOM,从而减少应用的包大小。

✅ 使用 Vapor Mode 的好处如下:

- 性能更佳
    
- 使用更少的内存
    
- 需要更少的运行时支持代码。
    

Solid.js 具有与 Vue 非常相似的响应式系统,它们都在其响应式系统中使用代理,

最终目标是在同一个应用中自由混合 Vapor 和非 Vapor 组件。

图片

注意:Vapor Mode 最初将仅使用 Vapor 树集成到 vDOM 中,并在此基础上逐步提高它们之间的互操作性。

通过这种方式编译的应用程序将能够完全删除虚拟 DOM 运行时,只包含极为轻量级的@vue/reactivity和 vapor mode 运行时辅助程序。与当前使用 vDOM 的 Vue 3 应用的基线大小(约 50KB)相比,这样编写应用的基线大小仅为约 6KB,实现了高达 88%的压缩比!

关键点

  1. 这是一项可选功能,不会影响现有的代码库。如果计划将 Vue 版本升级到包含 Vapor Mode 的版本,这不会引入任何与 Vapor Mode 相关的重大更改。

  2. Vapor Mode 将能够与 vDOM 组件进行互操作。因此,如果希望使用像 Vuetify 这样的虚拟 DOM 库, Vapor Mode 仍然能够支持它。

vue3.4 log

  • vue3.4 完全重写了模版解析器。 旧版本使用递归下降解析器,该解析器依赖了大量正则表达式和前向搜索。 新的解析器使用基于htmlparser2的状态机tokenizer(分词器),它仅迭代了整个模版字符串一次。预计对SFC单文件组件,提升了44%的速度。

  • 更好的响应系统,双向绑定 ref 双向绑定属性,只有当value发生切实变化时,才能出发watch和computer等方法。

  • 属性绑定同名简写

<img :id="id" :src="src" :alt="alt">
# 简写
<img :id :src :alt>

混合开发hybrid

  • 配合 ElectronTauri 构建桌面应用
  • 配合 Ionic Vue 构建移动端应用
  • 使用 Quasar 用同一套代码同时开发桌面端和移动端应用

小技巧

object响应式

image-20230118182843044

特殊知识

  • Suspense 提供改进的加载和错误处理机制异步组件;
  • Provide/Inject 简化了组件之间的数据传输和通信。
  • 渲染函数h(), 当你需要附加其他属性时, 会导致失去响应式. 坑啊

传送组件teleport

将插槽传送到外面dom树上

<template>
  <div>
    <!-- Other component -->
    <teleport to="body">
      <!-- Popup box content -->
    </teleport>
  </div>
</template>

多组件封装fragment

<template>
  <fragment>
    <h1>your title</h1>
    <p>this is a test content</p>
  </fragment>
</template>

定制指令

<template>
  <div>
    <input v-auto-focus />
  </div>
</template>

<script>
export default {
  directives: {
    autoFocus: {
      mounted(el) {
        el.focus();
      }
    }
  }
};
</script>