Vue 自学笔记

这篇笔记用于自己复习,知识点很散,不适合别人看

Vue 基础

应用实例 const app = Vue.createApp(RootOption)

根组件选项 const RootOption = {}
根组件实例 vm = app.mount('#app')

组件选项:与根组件的配置选项相同
组件实例:每个组件都有一个组件实例,vm 通常代表组件实例

组件选项

inject

将 provide/inject 看做是长距离的 props
默认情况下,provide/inject 绑定并不是响应式的

props

单向下行绑定,父组件中更新 props 属性值,则子组件中的值自动更新,然后自动更新视图
不要在子组件中更改 props 属性值
prop 会在一个组件实例创建之前进行验证,所以实例的 property (如 data、computed 等) 在 default 或 validator 函数中是不可用的
当 prop 验证失败的时候,(开发环境构建版本的) Vue 将会产生一个控制台的警告。

data

data 是个函数,返回一个对象,该对象作为 vm.data,具有响应式行为。 vm.count 与 vm.data.count 一样,可读写
vm.foo 动态添加的属性不具备响应式,该属性也不会保存在 vm.data,vm.data.foo undefined
相同组件的多个实例拥有各自不同的 $data 属性,类似对象的实例属性

computed

依赖的 data 发生更新,则 computed 自动更新,对应的视图自动更新

computed vs watch

这两个都可以根据一个属性的变化来改变另一个属性,大多数需要用到 computed,因为他可以缓存。
只有在某个属性改变之后需要进行异步操作或者开销较大的复杂操作的时候,需要使用 watch

methods

可以将 methods 方法作为事件的处理函数 v-on:click="increment"
可以从模板中直接调用 methods 中定义的方法 v-bind:title="getTitle()" {{ getMessage() }}
从模板中调用方法,如果方法访问了任何响应式数据,则将其作为渲染依赖项进行跟踪
从模板调用的方法不应该有任何副作用,比如更改数据或触发异步进程
methods 中的方法被相同组件的多个实例共享,类似对象的原型方法

setup

生命周期钩子

beforeCreated:已经通过 vm.$on() 为组件监听了所有事件,包括生命周期钩子
created:可以访问组件的所有注入属性以及响应式属性
beforeMounted: 已经将组件的模板字符串编译成了 render() 函数。
mounted: 组件的模板已经生成 DOM 并且插入到文档中了。
beforeUpdated: 响应式 data 已经发生了变化
updated: 组件的 DOM 视图已经发生了改变
beforeUnmounted: 已经调用了 vm.unmount()
unmounted: 组件 DOM 结构已经从文档中删除,组件响应式数据已经删除,组件事件监听器已经删除

组件实例属性

不管是哪种属性,都能够在该组件作用域的模板中直接访问

组件选项定义的属性

vm.count

内置属性

vm.data vm.attr
vm.emit vm.on

动态添加的属性

vm 实例创建之后动态添加的属性不能够响应式

vm.foo = 100
vm.$data.foo //undefined

组件

全局注册

app.component('my-component', {})

局部注册

{
component: {
'my-component': {}
}
}

prop 特性

非 prop 特性

自定义特性
class
style
事件

都会传递到子组件的根元素上,class style 与根元素的原有 class style 合并到一起。
如果子组件根元素不支持该事件,则无事发生 element.addEventListener('no-suposed-event', function () {}),则不会触发任何监听器。

通过 inheritAttrs: false 配合 v-bind="attrs",可以将非 prop 特性绑定到子组件的任何一个元素上,这里的attrs 包括 html特性、class、style、事件,与 vue2 不同,vue3 attrs 包括事件,而且不再保留listens 属性了

自定义事件

emits: ['myEvent', 'click']

当在 emits 选项中定义了原生事件 (如 click) 时,将使用组件中的自定义事件替代原生事件侦听器。
凡是在子组件中 $emit() 的事件,最好在 emits 属性定义该事件

props属性验证失败,在开发模式下,只是在浏览器控制台警告,不影响程序执行
自定义事件的验证如果失败,在开发模式下,只是在浏览器控制台警告,不影响程序执行,不影响发送事件以及处理事件

v-model

默认情况下,组件上的 v-model 使用 modelValue 作为 prop 和 update:modelValue 作为事件

<MyComponent v-model="user"></MyComponent>
<MyComponent :modelValue="user" @update:modelValue="user=$event"></MyComponent>

可以给 v-model 添加参数,来改变默认值,使用 person 作为 prop,使用 update:person 作为事件

<MyComponent v-model:person="user"></MyComponent>
<MyComponent :person="user" @update:person="user=$event"></MyComponent>

ref

<input ref="foo">
<my-component ref="bar"></my-component>

this.refs.foo 指向 Element 元素 this.refs.bar 指向子组件实例

Vue Router

路由

<router-link to="/"></router-link>
<router-view></router-view>

router-link 和 router-view 可以放在任何组件的任何位置,不是非要放在根组件里面,这两个甚至可以分开放在不同组件里。

router-link 本质上就是渲染 a 标签,放在哪个组件里都可以
router-view 就是渲染与 path 对应的组件,在哪渲染都可以

routes 是一个数组,每个元素是一个路由记录 RouteRecord 对象,定义路由与组件的映射关系

router 实例,定义了路由的方法

router 同时还是个插件,app.use(router) 可以插入全局实例属性 this.routes this.router

vue-router 最核心的工作就是编写 routes 映射关系

URL:

  • '/foo/bar'
  • { path: '/foo/bar', query: { id: 1 }, hash: '#c=1' }
  • { name: 'user', params: { username: 'lee' } }
<router-link :to="URL"></router-link>
<router-link :to="URL" replace></router-link>
router.push(URL)
router.replace(URL)
router.go(n)
{ path: '/:username*', component: User },
{ path: '/:pid(\\d+)+', component: Post },


{ path: '/foo/:bar' },
{ path: '/foo/:bar/baz/:tar' },

{ path: '/foo/:bar+' }, // $route.params.bar []
{ path: '/foo/:bar*' }, // $route.params.bar []
{ path: '/foo/:bar?' },

{ path: '/foo/:bar(\\d+)+' },
{ path: '/foo/:bar(\\d+)*' },
{ path: '/foo/:bar(\\d+)?' }

嵌套路由

const routes = [
  {
    path: '/user/:username',
    component: User,
    children: [
      {
        path: 'profile',
        component: UserProfile
      },
      {
        path: 'posts',
        component: UserPosts
      },
      {
        path: '',
        component: UserHome
      }
    ]
  }
]

顶级 <router-view> 匹配顶层 path 渲染,
组件 User 中的 <router-view> 匹配 User Route 的 children 中的 path 渲染

命名视图

一个路由映射一组组件 { path: '', components: {} }

为一个匹配路径提供多个 <router-view>,来渲染多个组件,通过 <router-view name=""> name 属性来指定对应的 <router-view> 应该渲染哪个组件

const routes = [
  {
    path: '/settings',
    component: UserSettings,
    children: [
      {
        path: 'email',
        component: UserEmail
      },
      {
        path: 'profile',
        components: {
          default: UserProfile,
          helper: UserProfileView
        }
      }
    ]
  }
]

重定向和别名



{ path: '/home', redirect: '/' }

重定向改变浏览器中的 url
重定向不需要映射 component

{ path: '/', alias: '/home', component: HomePage }

别名不改变浏览器中的 url

可以在嵌套路由里面提供别名

``` js
{
  path: '/users',
  component: UsersLayout,
  children: [
    // 为这 3 个 URL 呈现 UserList
    // - /users
    // - /users/list
    // - /people
    { path: '', component: UserList, alias: ['/people', 'list'] },
  ],
}

如果路由有参数,绝对路径的别名必须带着参数

{
  path: '/users/:id',
  component: UsersByIdLayout,
  children: [
    // 为这 3 个 URL 呈现 UserDetails
    // - /users/24
    // - /users/24/profile
    // - /24
    { path: 'profile', component: UserDetails, alias: ['/:id', ''] },
  ],
}

路由组件传参

{{ username }} 取代 {{ $route.params.username }}
{ path: '/user/:username', component: User, props: true }

props 为 true,route.params 将被设置为组件 User 的 props
props 可以是对象 { title: 'hello', id: 20 },传递静态 props 给组件 User

导航守卫

路由元信息

{ path: '', component: {}, meta: { foo: 100, bar: 'hello' } }

to.meta
to.matched.some(record => record.meta.foo === 100)

数据获取

导航完成后,在组件的 created 钩子函数中从服务器获取数据
导航完成前,在组件的 beforeRouteEnter 导航守卫中从服务器获取数据

Vue 动画

<Transition> 组件

<Transition> 是一个内置组件,这意味着它在任意别的组件中都可以被使用,无需注册。
<Transition> 仅支持单个元素或组件作为其插槽内容。如果内容是一个组件,这个组件必须仅有一个根元素。
在位于 <Transition></Transition> 之间的单个元素或组件进入或者离开 DOM 时候,会为该元素或组件应用动画

进入或离开可以由以下的条件之一触发:

  • 由 v-if 所带来的条件渲染
  • 由 v-show 所带来的条件显示
  • 由特殊元素 <component :is=''> 切换的动态组件

当一个 <Transition> 组件中的元素被插入或移除时,会发生下面这些事情:

  1. Vue 会自动检测目标元素是否应用了 CSS 过渡或动画。如果是,则一些 CSS 过渡 class 会在适当的时机被添加和移除。

  2. 如果有作为监听器的 JavaScript 钩子,这些钩子函数会在适当时机被调用。

  3. 如果没有探测到 CSS 过渡或动画、没有提供 JavaScript 钩子,那么 DOM 的插入、删除操作将在浏览器的下一个动画帧上执行。

自动添加 class

.v-enter-from
.v-enter-active
.v-enter-to

.v-leave-from
.v-leave-active
.v-leave-to

<Transition name="fade"></Transition>

.fade-enter-actvie
.fade-leave-active
...

可以配合 Animate.css 使用

<Transition
  enter-active-class="animate__animated animate__tada"
  leave-active-class="animate__animated animate__bounceOutRight"
>
  <p v-if="show">hello</p>
</Transition>

CSS transition

.v-enter-active {
transition: all 0.3s ease-out;
}
.v-leave-active {
transition: all 0.8s cubic-bezier(1, 0.5, 0.8, 1);
}

.v-enter-from,
.v-leave-to {
transform: translateX(20px);
opacity: 0;
}

CSS animation

.v-enter-active {
animation: bounce-in 0.5s;
}
.v-leave-active {
animation: bounce-in 0.5s reverse;
}

@keyframes bounce-in {
0% {
transform: scale(0);
}
50% {
transform: scale(1.25);
}
100% {
transform: scale(1);
}
}

高级

<Transition type="animation"></Transition>

<Transition> 组件会通过监听过渡根元素上的第一个 transitionend 或者 animationend 事件来尝试自动判断过渡何时结束。如果根元素上同时使用了 css 过渡和 css 动画,则通过 type 指定监听哪个事件。duration 属性可以显式指定过渡的持续时间 (以毫秒为单位),而不是通过监听 transitionend 或者 animationend 来决定过渡何时结束。

多用 transform 和 opacity,不要使用 margin height 这类会影响 CSS 布局导致 DOM 重绘的属性。

JavaScript 钩子

<Transition
  @before-enter="onBeforeEnter"
  @enter="onEnter"
  @after-enter="onAfterEnter"
  @enter-cancelled="onEnterCancelled"
  @before-leave="onBeforeLeave"
  @leave="onLeave"
  @after-leave="onAfterLeave"
  @leave-cancelled="onLeaveCancelled"
>
  <!-- ... -->
</Transition>

总结

<Transition
  appear
  type="animation|transition"
  :css="false"
  :duration="550"
  mode="out-in"

  name="fade"
  enter-from-class=""
  enter-active-class=""
  enter-to-class=""
  leave-from-class=""
  leave-active-class=""
  leave-to-class=""

  @before-enter="onBeforeEnter"
  @enter="onEnter"
  @after-enter="onAfterEnter"
  @enter-cancelled="onEnterCancelled"
  @before-leave="onBeforeLeave"
  @leave="onLeave"
  @after-leave="onAfterLeave"
  @leave-cancelled="onLeaveCancelled"
>
  <p v-if="show">hello</p>
</Transition>

动态过渡

对 <Transition> 的 prop 的值进行动态绑定 <Transition :name="transitionName" :duration="transitionDuration">,根据当前组件的 data 的值来决定过渡行为

可重用

<template>
  <!-- 包装内置的 Transition 组件 -->
  <Transition
    name="my-transition"
    @enter="onEnter"
    @leave="onLeave">
    <slot></slot> <!-- 向内传递插槽内容 -->
  </Transition>
</template>

<style>
/*
  必要的 CSS...
  注意:避免在这里使用 <style scoped>
  因为那不会应用到插槽内容上
*/
</style>
<MyTransition>
  <div v-if="show">Hello</div>
</MyTransition>

<TransitionGroup> 组件

<TransitionGroup> 是一个内置组件,设计用于呈现一个列表中的元素或组件的插入、移除和顺序改变的动画效果。

  • 默认情况下,它不会渲染一个包装器元素。但你可以通过传入 tag prop 来指定一个元素作为包装器元素来渲染。
  • 过渡模式 mode 属性在这里不可用,因为我们不再是在互斥的元素之间进行切换。
  • 其中的元素总是必须有一个独一无二的 key attribute。
  • CSS 过渡 class 会被应用在其中的每一个元素上,而不是这个组的容器上。

.v-move {
transition: all 0.5s ease;
}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容