搭建vue3组件库(三): CSS架构之BEM

server/2024/9/24 14:12:39/

文章目录

    • 1. 通过 JS 生成 BEM 规范名称
      • 1.1 初始化 hooks 目录
      • 1.2 创建 BEM 命名空间函数
      • 1.3 通过 SCSS 生成 BEM 规范样式
    • 2. 测试 BEM 规范

BEM 是由 Yandex 团队提出的一种 CSS 命名方法论,即 Block(块)、Element(元素)和 Modifier(修改器)的简称,是 OOCSS 方法论的一种实现模式,底层仍然是面向对象的思想。

  • BEM 规范下 classname 的命名格式为:
    • 所有实体的命名均使用小写字母,复合词使用连字符 “-” 连接。
    • Block 与 Element 之间使用双下画线 “__” 连接。
    • Mofifier 与 Block/Element 使用双连接符 “–” 连接。
    • modifier-name 和 modifier_value 之间使用单下画线 “_” 连接。

1. 通过 JS 生成 BEM 规范名称

在编写组件的时候如果通过手写 classname 的名称,那么需要经常写 - 、 __ 、 --,那么就会变得非常繁琐,BEM 命名规范是具有一定规律性的,所以可以通过 JavaScript 按照 BEM 命名规范进行动态生成。

1.1 初始化 hooks 目录

packages 目录下创建一个 hooks 目录,进入到 hooks 目录底下初始化一个 package.json 文件,修改内容如下:

{"name": "@vision-ui-vue/hooks","version": "0.0.1","license": "MIT","main": "index.ts","module": "index.ts","unpkg": "index.js","jsdelivr": "index.js","types": "index.d.ts","peerDependencies": {"vue": "^3.2.0"}
}

1.2 创建 BEM 命名空间函数

hooks 目录下再创建一个 use-namespace 目录用于创建 BEM 命名空间函数,再在 hooks 目录下创建一个 index.ts 文件用于模块入口文件。

// index.ts
import * from './use-namespace'
// use-namespace/index.ts
import { computed, unref } from 'vue'// 默认的命名空间
export const defaultNamespace = 'v'
// 状态前缀
const statePrefix = 'is-'/*** BEM 命名字符拼接函数* @param namespace 命名空间* @param block 块名* @param blockSuffix 块的后缀* @param element 元素名* @param modifier 修改器名* @returns 拼接后的BEM类名字符串*/
const _bem = (namespace: string,block: string,blockSuffix: string,element: string,modifier: string
) => {// 默认是 Blocklet cls = `${namespace}-${block}`// 如果存在 Block 后缀,也就是 Block 里面还有 Blockif (blockSuffix) {cls += `-${blockSuffix}`}// 如果存在元素if (element) {cls += `__${element}`}// 如果存在修改器if (modifier) {cls += `--${modifier}`}return cls
}/*** 用于创建和管理BEM类名的工具函数* @param block 块名* @returns 返回一个对象,包含用于创建BEM类名的各种方法*/
export const useNamespace = (block: string) => {// 基于Vue的computed创建动态命名空间const namespace = computed(() => defaultNamespace)// 创建块级类名 v-formconst b = (blockSuffix = '') =>_bem(unref(namespace), block, blockSuffix, '', '')// 创建元素级类名 v-input__innerconst e = (element?: string) =>element ? _bem(unref(namespace), block, '', element, '') : ''// 创建修改器类名 v-form--defaultconst m = (modifier?: string) =>modifier ? _bem(unref(namespace), block, '', '', modifier) : ''// 创建带后缀的块元素类名 v-form-itemconst be = (blockSuffix?: string, element?: string) =>blockSuffix && element? _bem(unref(namespace), block, blockSuffix, element, ''): ''// 创建元素的修改器类名 v-scrollbar__wrap--hidden-defaultconst em = (element?: string, modifier?: string) =>element && modifier? _bem(unref(namespace), block, '', element, modifier): ''// 创建块后缀的修改器类名 v-form-item--defaultconst bm = (blockSuffix?: string, modifier?: string) =>blockSuffix && modifier? _bem(unref(namespace), block, blockSuffix, '', modifier): ''// 创建块元素的修改器类名 v-form-item__content--xxxconst bem = (blockSuffix?: string, element?: string, modifier?: string) =>blockSuffix && element && modifier? _bem(unref(namespace), block, blockSuffix, element, modifier): ''// 创建动作状态类名,支持两种调用方式const is: {(name: string, state: boolean | undefined): string(name: string): string} = (name: string, ...args: [boolean | undefined] | []) => {const state = args.length >= 1 ? args[0]! : truereturn name && state ? `${statePrefix}${name}` : ''}return {namespace,b,e,m,be,em,bm,bem,is,}
}

1.3 通过 SCSS 生成 BEM 规范样式

packages/theme-chalk 目录下创建一个 src 目录,在 src 目录下创建一个 mixins 目录。在 mixins 目录下新建件:config.scssmixins.scssfunction.scss, config.scss 编写 BEM 的基础配置比如样式名前缀、元素、修饰符、状态前缀:

$namespace: 'v' !default; // 所有的组件以v开头,如 v-input
$common-separator: '-' !default; // 公共的连接符
$element-separator: '__' !default; // 元素以__分割,如 v-input__inner
$modifier-separator: '--' !default; // 修饰符以--分割,如 v-input--mini
$state-prefix: 'is-' !default; // 状态以is-开头,如 is-disabled
// 在 SCSS 中,使用 $+ 变量名:变量 来定义一个变量。在变量后加入 !default 表示默认值。给一个未通过 !default 声明赋值的变量赋值,此时,如果变量已经被赋值,不会再被重新赋值;但是如果变量还没有被赋值,则会被赋予新的值。

mixins.scss 编写 SCSS 的 @mixin 指令定义的 BEM 代码规范:

@use '../common/var' as *;// 定义 Block
@mixin b($block) {$B: $namespace + $common-separator + $block !global;.#{$B} {@content;}
}// 定义 Element
@mixin e($element) {$E: $element !global;$selector: &;$currentSelector: '';@each $unit in $element {$currentSelector: #{$currentSelector +'.' +$B +$element-separator +$unit +','};}@if hitAllSpecialNestRule($selector) {@at-root {#{$selector} {#{$currentSelector} {@content;}}}} @else {@at-root {#{$currentSelector} {@content;}}}
}// 定义修改器
@mixin m($modifier) {$selector: &;$currentSelector: '';@each $unit in $modifier {$currentSelector: #{$currentSelector +$selector +$modifier-separator +$unit +','};}@at-root {#{$currentSelector} {@content;}}
}// 定义动作状态
@mixin when($state) {@at-root {&.#{$state-prefix + $state} {@content;}}
}

function.scss 定义一些 SCSS 的 @function 指令定义的函数:

@use 'config';// 该函数将选择器转化为字符串,并截取指定位置的字符
@function selectorToString($selector) {$selector: inspect($selector); // inspect(...) 表达式中的内容如果是正常会返回对应的内容,如果发生错误则会弹出一个错误提示。$selector: str-slice($selector, 2, -2); // str-slice 截取指定字符@return $selector;
}
// 判断父级选择器是否包含'--'
@function containsModifier($selector) {$selector: selectorToString($selector);@if str-index($selector, config.$modifier-separator) {// str-index 返回字符串的第一个索引@return true;} @else {@return false;}
}
// 判断父级选择器是否包含'.is-'
@function containWhenFlag($selector) {$selector: selectorToString($selector);@if str-index($selector, '.' + config.$state-prefix) {@return true;} @else {@return false;}
}
// 判断父级是否包含 ':' (用于判断伪类和伪元素)
@function containPseudoClass($selector) {$selector: selectorToString($selector);@if str-index($selector, ':') {@return true;} @else {@return false;}
}
// 判断父级选择器,是否包含`--` `.is-`  `:`这三种字符
@function hitAllSpecialNestRule($selector) {@return containsModifier($selector) or containWhenFlag($selector) orcontainPseudoClass($selector);
}

2. 测试 BEM 规范

  • 在根目录执行 pnpm install sass -D -w
  • 接着执行 pnpm install @vision-ui-vue/hooks -D -w 把 hooks 引入到项目中
  • packages/components 下新建 button 目录,目录结构为
├── packages
│   ├── components
│   │   ├── button
│   │   │   ├── src
│   │   │   │   └── button.vue
│   │   │   └── index.ts
│   │   └── package.json

index.ts 内容:

import Button from './src/button.vue'
export default Button

button.vue 内容:

<template><button :clsaa="bem.b()">测试按钮</button>
</template><script lang="ts" setup>
import { useNamespace } from '@vision-ui-vue/hooks'
const bem = useNamespace('button')
</script>
  • 进入 play 项目,修改 App.vue 文件,引入 button 组件:
<template><div><v-button></v-button></div>
</template><script setup lang="ts">
import VButton from '@vision-ui-vue/components/button'
</script>
  • 运行 play 项目,查看效果
    打开控制台,可以看到按钮的class为 v-button
    在这里插入图片描述

http://www.ppmy.cn/server/28744.html

相关文章

什么是RabbitMQ,RabbitMQ基本概念,RabbitMQ的使用场景

目录 面试官:什么是RabbitMQ,RabbitMQ的使用场景什么是RabbitMQ?RabbitMQ基本概念RabbitMQ的使用场景举例该文章专注于面试,面试只要回答关键点即可,不需要对框架有非常深入的回答,如果你想应付面试,是足够了,抓住关键点 面试官:什么是RabbitMQ,RabbitMQ的使用场景 …

OpenHarmony实战开发-动画曲线、如何实现动画衔接

UI界面除了运行动画之外&#xff0c;还承载着与用户进行实时交互的功能。当用户行为根据意图变化发生改变时&#xff0c;UI界面应做到即时响应。例如用户在应用启动过程中&#xff0c;上滑退出&#xff0c;那么启动动画应该立即过渡到退出动画&#xff0c;而不应该等启动动画完…

PG数据库结构与oracle比较

1.数据库集簇逻辑结构 数据库集簇概念&#xff1a;一个大的数据库是由若干个小的数据库组成&#xff0c;实现数据的隔离存放&#xff0c;在概念上应该是与mysql一样的 在mysql中可以用show database列出数据库 PG中用\l 数据库对象存放在数据库中&#xff1a; PG中的所有数据…

cve-2018-19518漏洞复现

一、靶场的启动 在相应的文件夹位置打开终端后进行如下操作 1.运行此靶场 sudo docker-compose up -d 2.查看启动环境 sudo docker ps 3.关闭此靶场环境 docker-compose down 二、漏洞内容简介 php imap扩展用户在php中执行邮件收发操作&#xff0c;其imap_open函数会调用rsh…

微信小程序常用的api

基础API&#xff1a; wx.request&#xff1a;用于发起网络请求&#xff0c;支持GET、POST等方式&#xff0c;是获取网络数据的主要手段。wx.showToast&#xff1a;显示消息提示框&#xff0c;通常用于向用户展示操作成功、失败或加载中等状态。wx.showModal&#xff1a;显示模态…

【问题分析】TaskDisplayArea被隐藏导致的黑屏以及无焦点窗口问题【Android 14】

1 问题描述 用户操作出的偶现的黑屏以及无焦点窗口问题。 直接原因是&#xff0c;TaskDisplayArea被添加了eLayerHidden标志位&#xff0c;导致所有App的窗口不可见&#xff0c;从而出现黑屏和无焦点窗口问题&#xff0c;相关log为&#xff1a; 这个log是MTK添加的&#xff0…

File contains parsing errors: file:///etc/yum.repos.d/nginx.repo报错解决,文件配置出现问题

执行yum指令出现以下错误&#xff1a; 解决方案&#xff1a;yum的配置文件出现问题&#xff0c; 先删除yum.repos.d目录下所有文件 rm -f /etc/yum.repos.d/* 然后重新下载阿里的资源 wget -O /etc/yum.repos.d/CentOS-Base.repo http://mirrors.aliyun.com/repo/Centos-7.…

【Web前端】Javascript冒泡排序、选择排序、插入排序

一、冒泡排序 简单说一下冒泡排序过程&#xff0c;一个列表中有n个可排序元素&#xff0c;如果按升序&#xff08;降序&#xff09;排列&#xff0c;就需要进行n-1次循环完成排序&#xff08;就像魔方拼好5个面自然就完全拼好了&#xff09;&#xff0c;每次循环按次两两比较&a…