首页 > web前端 > js教程 > 正文

Vue中SFC和vue-loader的具体分析

不言
发布: 2018-07-21 11:19:35
原创
5425人浏览过

本篇文章给大家分享的是关于vue中sfc和vue-loader的具体分析,内容很不错,有需要的朋友可以参考一下,希望可以帮助到大家。

官网:https://vue-loader.vuejs.org/zh/
vue-loader是一个webpack的loader,简单来说是将vue文件转换为JS模块。在监测到babel-loader或者buble-loader配置时,自动支持ES2015;
使用vue-loader就可以用Vue Single-File Component (SFC)即单文件组件的形式编写一个组件。

.vue单文件组件 (SFC) 规范

1.

一个SFC中最多一个块;
其内容将被提取为字符串传递给 vue-template-compiler ,然后webpack将其编译为js渲染函数,并最终注入到从 <script> 导出的组件中;</script>

2. <script>脚本块</script>

一个SFC最多一个<script>块;<br/>它的默认导出应该是一个 Vue.js 的组件选项对象,也可以导出由 Vue.extend() 创建的扩展对象。<br/>思考:Vue.extend() 中 data 必须是函数,所以在.vue SFC的script中,export中的data是函数</script>

3.

一个 .vue 文件可以包含多个

立即学习前端免费学习笔记(深入)”;

4. 自定义语言块

vue-loader 将会使用块名来查找对应的  loader 进行处理,需要配置webpack.config

5. 所有语言块支持src导入

导入路径遵循和 webpack 模块请求相同的路径解析规则
//  相对路径需要以../开始


vue-loader特性概述

  • 支持组件的各个template,script,style模块使用其他非默认语言,比如:

  • 除了默认的

  • 为组件中的css模拟局部作用域,直接

  • 将style和template中的静态资源(图片等)当作模块依赖,并通过webpack loader处理

  • 在开发模式下的热重载

预处理

Vue支持各类型的预处理器,这些预处理可以编译语言块,例如Vue默认使用PostCSS处理css,你可以用sass,less,stylus等, 对于js部分Vue默认使用babel处理,还可以用coffee、typescript等,只需要安装相应的loader加载器,Vue会根据语言块的lang属性和webpack配置的option rules推断对应的loader。

小栗子:css使用sass预处理

$ npm install sass-loader node-sass  --save-dev
登录后复制
// webpack.config.js -> module.rules
{
    test: /\.sass$/,
    use: [
        'vue-style-loader',
        'css-loader',
        {
            loader: 'sass-loader',
            options: {
                indentedSyntax: true  //sass-loader 默认解析 SCSS 语言
            }
        }
    ]
}
登录后复制
<!--  .vue -> style 增加lang属性并赋值 -->
<style lang="sass">
/* write SASS here */
</style>
登录后复制
对于 模版
<!--  .vue -> template -->
<template lang="pug">
p
  h1 Hello world!
</template>
登录后复制

CSS作用域:scope

<style>
/* global styles */
</style>

<style scoped>
/* local styles */
.example {
  color: red;
}
</style>
<template>
  <p class="example">hi</p>
</template>
登录后复制

vue-loader处理的CSS输出,都是通过PostCSS进行作用域重写,PostCSS处理后如下:

<style>
.example[data-v-f3f3eg9] {
  color: red;
}
</style>

<template>
  <p class="example" data-v-f3f3eg9>hi</p>
</template>
登录后复制
  • 使用scope时,子组件的根节点将受父组件作用域CSS影响

使用scope作用域时,父组件的样式不会泄漏到子组件中。 但子组件的根节点将受父级作用域CSS和子级作用域CSS的影响。 这是为了父级可以设置子组件根元素的样式以进行布局。

  • 使父组件可以使用‘ >>>  ’或‘ /deep/  ’ 这种深度选择器作用于子组件
<style scoped>
.a >>> .b { /* ... */ }
</style>
登录后复制

编译为

.a[data-v-f3f3eg9] .b { /* ... */ }
登录后复制
  • 动态生成的DOM内容不受scope style的影响,但可以使用深度选择器进行样式改变
  • 使用scope作用域不能弃用class或id等

考虑到浏览器渲染各种 CSS 选择器的方式,当使用 scoped 时,选择属性选择器如p { color: red } 在作用域中会慢很多倍(即当与属性选择器组合时)。如果你使用 class或者 id 代替,比如 .example { color: red },这样几乎没有性能影响

  • 在递归组件中注意后代选择器

对于带有.a .b的CSS选择器,如果匹配.a的元素包含递归子组件,则该子组件中的所有.b将与其匹配。

小思考:在template中只包含一个外层节点,不能多个节点并列,这个设计思路遵循父组件可以操作子节点的一个根节点,即使在CSS局部作用域下依然有效

CSS模块模式:module

一个CSS Module其实就是一个CSS类型的文件,其编写方式与CSS相同,但在编译时会编译为ICSS低级交换格式。
其默认所有的类名/动画名都在本地作用域,当从JS模块导入CSS模块时,它会导出包含从本地名称到全局名称的所有映射的一个对象

在CSS Module中,所有的url和@import都是被看成模块依赖,例如url(./image.png) 会被转换为 require(‘./image.png’)
CSS 模块处理是通过 css-loader,请求的资源可以是在node_modules中。
Vue loader中的CSS  Module

配置

Vue loader集成了CSS Module,使用前需要在css-loader中设置modules: true,如下:

// webpack.config.js -> module.rules
{
  test: /\.css$/,
  use :[
    'vue-style-loader',
    {
      loader: 'css-loader',
      options: {
        // enable CSS Modules
        modules: true,
        // customize generated class names
        localIdentName: '[local]_[hash:base64:8]'
      }
    }
  ]
}
登录后复制

然后就在

<style module>
.red {
  color: red;
}
</style>
登录后复制

使用

这样Vue loader会将css module本地对象编译为计算属性注入到组件中,默认值为$style。template可以通过动态类绑定使用:

<template>
  <p :class="$style.red">
    This should be red
  </p>
</template>
登录后复制

JS也可以通过this.$style访问:

<script>
export default {
  created () {
    console.log(this.$style.red)
    // -> "red_1VyoJ-uZ"
    // an identifier generated based on filename and className.
  }
}
</script>
登录后复制

为多个style自定义标识符

一个.vue组件可以有多个

<style module="a">
  /* 标识符注入为 a */
</style>
<style module="b">
  /* 注入为 b */
</style>
登录后复制

结合预处理

CSS Module还可以在预处理(SASS,LESS等)中使用,在配置webpack rules时加上modules: true,例如:

// webpack.config.js -> module.rules test: /\.scss$/ 
use[{
      loader: 'css-loader',
      options: { modules: true }
 }]
登录后复制

.vue中自定义节点/语言块

除了默认的

小栗子:自定义语言块

首先需要loader。为了将自定义块注入到组件中,自定义一个loader如下:

// myblock-loader.js 
module.exports = function (source, map) {
  this.callback(
    null,
    `export default function (Component) {
      Component.options.__myblock = ${
        JSON.stringify(source)
      }
    }`,
    map
  )
}
登录后复制

配置到webpack

// webpack.config.js -> module.rules
{
    resourceQuery: /blockType=myblock/,
    loader: require.resolve('./myblock-loader.js')
}
登录后复制

在组件中使用

<!-- ComponentB.vue -->
<template>
  <p>Hello</p>
</template>

<myblock>
This is the documentation for component B.
</myblock>
登录后复制

组件间的复用

<!-- ComponentA.vue -->
<template>
  <p>
    <ComponentB/>
    <p>{{ myblock }}</p>
  </p>
</template>

<script>
import ComponentB from './ComponentB.vue';

export default {
  components: { ComponentB },
  data () {
    return {
      myblock: ComponentB.__myblock
    }
  }
}
</script>
登录后复制

热重载:不刷新页面的更新

当更改.vue的template或style时,页面不刷新也可实现内容动态的变化,并保留着程序及其组件的状态。在以下情况下会保持热重载:

  1. 编辑组件的

  2. 编辑组件的

  3. 编辑组件的部分<script>,编辑的实例将销毁后重建,这是因为<script>可能包含产生副作用的生命周期钩子,因此需要“重新加载”以确保一致的行为。如果组件产生全局副作用则需要整页的重新加载。</script>

热加载是默认启动的,当以下情况下关闭:

  • webpack target is node (SSR)

  • webpack压缩代码

  • 生产模式:process.env.NODE_ENV === 'production'

也可以手动禁用热重载:

// webpack.config.js -> module.rules
{
    test: /\.vue$/,
    loader: 'vue-loader',
    options: {
        hotReload: false // disables Hot Reload
    }
}
登录后复制

功能组件

一个简单的.vue组件可以声明为无状态的(没有反应数据)和无实例的(没有此上下文)的功能组件。

vue.js 2.5版本以上,在template中加上functional属性即可声明:

<template functional>

</template>
登录后复制

在script中的结构为:

Vue.component('my-component', {
  functional: true,
  props: { //也可不定义props
    // ...
  },
  render: function (createElement, context) {
    // createElement, context 为上下文参数
    // ...
  }
})
登录后复制

组件加上functional: true后,更新渲染函数需要添加context参数,故this.$slots.default更新为context.children,this.level更新为context.props.level。
功能组件通过context对象传递/获取所有的内容,例如:

  • context.props: 包含功能组件的props的对象,在vue 2.3+如未定义的props属性会自动加到组件根元素上;

  • context.children:  包含VNode子节点的数组

  • context.slots: 返回slots对象的函数

  • context.data: 整个数据对象,作为createElement的第二个参数传递给组件

  • context.parent: 指向父组件

  • context.listeners: (2.3.0+) data.on的别名,包含父级注册事件侦听器的对象。

  • context.injections: (2.3.0+) 包含组件的注入项

由于功能组件只是功能,缺少持久化实例,因此渲染的成本要低得多,也不会出现在Vue devtools组件树中;
适用于作为封装组件封装子节点/props/data后传递给子组件;
功能组件也支持预处理/scope/热重载等。

静态资源url是如何处理的

webpack配置module.exports需要vue-loader,同时也会有各种静态资源的loader。当Vue Loader编译SFC中的

@@##@@
登录后复制

会编译为:

createElement('img', {
  attrs: {
    src: require('../image.png') // this is now a module request
  }
})
登录后复制

URL路径解析规则:

  1. 绝对路径原样保存;

  2. 以“.”开头,则将其解释为相对模块请求,并根据文件系统上的文件夹结构解析;

  3. 以“~”开头,则将其后的内容解析为模块请求,可以在节点模块引用这些内容;

  4. 以“@”开头,则其后的内容也被解释为模块请求,@在vue-cli创建的项目中默认指向/ src,可以使用webpack配置@别名

相关推荐:

vue如何利用树形控件z-tree动态添加数据

Vue中SFC和vue-loader的具体分析

以上就是Vue中SFC和vue-loader的具体分析的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
作者最新文章
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习
PHP中文网抖音号
发现有趣的

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号