HTML5-类库系列 类名的各种操作

HTML5学堂:最近讲师团队在学习JS类库的知识,因此就跟大家一起共享一下类库的搭建吧。今天要讲解的功能是:类名的各种操作。在这里,HTML5学堂提醒各位,不要心急,各种复杂的内容都是从简单的内容一步步调整优化过来的。今天我们在前一篇的基础上来优化我们的功能。

第一步 增加类名 addClass

增加类名应该说是这三种功能当中最简单的一个,类名是一个字符串,只需要使用字符串的连接就能够实现这个功能。

function addClass(obj, newClassName) {
return obj.className += " " + newClassName;
}

我们定义一个功能函数,函数名称为addClass,需要为某个元素增加某个类名,因此这里采用两个参数,作为这个函数的接口。通过obj.className获取到当前该元素的类名,然后在其基础之上,与新类名使用字符串进行连接。注意两种类名之间需要有一个空格。这时候,细心的同学会发现,如果该元素初始没有类名,使用函数处理之后,会在最前面多一个空格。这个地方个人认为没有必要处理,因为它对我们的任何操作是没有什么影响的。

第二步 移除类名 removeClass

移除类名的功能函数就没有增加类名那么简单了~需要使用数组以及字符串等各种方法进行实现。当然也可以通过正则去实现。关于正则的方法,在下面的替换类名部分我们再讲,这里我们重点讲字符串与数组的处理方法。

移除类名的基本原理是:首先我们将元素的类名进行拆分,将这种class="HTML5 lili test",一个元素多个类名的内容,根据空格进行拆分,拆分并放置于数组当中。之后我们将数组中的类名分别和需要移除的类名进行比较,如果相同则删除掉。当所有的类名均比较完成之后,我们使用数组的join方法,将修改后的数组再拼合回字符串,赋值给元素的className(类名)即可。

具体实现代码:

function removeClass(obj, removeClassName) {
var classArr = obj.className.split(/\s+/);
var len = classArr.length;
for (var i = 0; i < len; i++) {
if (classArr[i] == removeClassName) {
delete classArr[i];
};
};
obj.className = classArr.join(" ");
}

第三步 替换类名

替换类名这个部分,看上去很简单,包括我自己在进行操作的时候,一开始也没有注意存在的问题。如果使用正则我们发现相对要简单很多,只需要使用replace方法即可。

function replaceClass(obj, newClassName, oldClassName) {
obj.className = obj.className.replace(oldClassName, newClassName);
}

在这里,我们传递了三个参数,分别是要修改类名的元素、需要替换的旧类名、新类名。使用replace方法,直接进行实现。实现之后自己进行了测试,发现在一种情况下是存在问题的。如下的测试用例,我们希望使用new替换html,按照我们的期望,是将class="wrap html5 html"替换为class="wrap html5 new",但是执行的结果反而是class="wrap new5 html"。

<!doctype html>
<html>
<head>
<meta charset="UTF-8">
<title>H5course - 刘国利 独行冰海</title>
<link rel="stylesheet" href="../css/reset.css">
<script src="../js/core.js"></script>
</head>
<body>
<div>
<p>HTML5 - 类库构建</p>
<p id="con" class="wrap html5 html">独行冰海 - 利利 - 刘国利</p>
<p class="wrap">梦幻雪冰 - 陈能堡</p>
</div>
<script>
replaceClass(con, "new", "html");
</script>
</body>
</html>

造成上述问题的原因是,正则匹配上的问题。我们需要确定是new为开头并且也是以new为结尾,才正常匹配。换句话说,就是使用/\bhtml\b/能够实现合理的匹配。

问题又来了,我们的html是从外部传入到函数内的,html实际上是oldClassName这个变量的变量值。那么我们能不能把\bhtml\b改成\boldClassName\b呢?很明显不现实的。想解决这个问题,我们先得知道如何在正则当中放置变量。

我们生成正则的方法有两种,一种是使用正则字面量的方式进行生成——/正则内容/,另一种是使用构造函数进行生成。使用正则字面量的方法生成时,会发现无论怎么添加变量,变量都会被当做正则的一部分。那么此时,使用第二种方法,正则的构造函数,就能够解决这个问题了。var reg = new RegExp("\\b"+oldClassName+"\\b"); 注意一点:反斜杠需要进行转义。

基本问题解决了,一起来看看我们的功能代码吧~

function replaceClass(obj, newClassName, oldClassName) {
var reg = new RegExp("\\b"+oldClassName+"\\b");
obj.className = obj.className.replace(reg, newClassName);
}

当然,还有一种方法能够实现,使用eval,不过,这种方法但凡学过AJAX部分的同学都知道,会影响JS性能。不过,在这里我还是给出eval的写法:var reg = eval("\\b"+oldClassName+"\\b");

最后,我们解决我们之前遗留的一个问题,如何用正则的方法实现类名的移除?巧妙的使用replace即可,我们只需要将newClassName换成""(空字符串)即可。

于是~完整版代码新鲜出炉喽~~~

/*
* 类名处理 增加、删除、替换
* 替换和移除都使用replaceClass进行控制
* 作者:独行冰海 - 利利
*/
function addClass(obj, newClassName) {
return obj.className += " " + newClassName;
}
function replaceClass(obj, oldClassName, newClassName) {
var reg = new RegExp("\\b"+oldClassName+"\\b");
if (newClassName) {
obj.className = obj.className.replace(reg, newClassName);
} else {
obj.className = obj.className.replace(reg, "");
}
}

原文发布于微信公众号 - HTML5学堂(h5course-com)

原文发表时间:2015-12-14

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏小詹同学

Leetcode打卡 | No.014 最长公共前缀

欢迎和小詹一起定期刷leetcode,每周一和周五更新一题,每一题都吃透,欢迎一题多解,寻找最优解!这个记录帖哪怕只有一个读者,小詹也会坚持刷下去的!

642
来自专栏IT派

Java面试中常被问到的几大技术难题

大家在平常面试java的过程中都会遇到哪些难题呢?还有一些即将去面试java的童鞋们,你们想知道技术面试中会涉及到哪些点吗?达妹为你整理Java面试中会被问到的...

1010
来自专栏有趣的Python

py编程技巧-1.1-如何在列表、字典、集合中根据条件筛选数据

所有代码先加这行解决中文乱码问题 # -*- coding: utf-8 -*- 版本区别注意: py2.7: xrange(); py3.x: ra...

3407
来自专栏肖洒的博客

前端入门学习--JavaScript

大概了解了HTML和CSS,到了前端的精华JavaScript。 学习笔记,ALL FROM 廖雪峰的官方网站

492
来自专栏PHP技术

PHP中正则表达式学习及应用

正则表达式元字符 * 匹配前一个内容的0次1次或多次 . 匹配内容的0次1次或多次,但不包含回车换行 + 匹配前一个内容的1次或多次 ?匹配前一个内容...

3038
来自专栏日常学python

爬虫必学知识之正则表达式下篇

这是日常学python的第13篇原创文章 继上篇文章说了正则表达式的简单用法,那今天我们就继续说一下正则表达式的复杂的用法。好了,废话不多说,直接进入正题。 正...

3747
来自专栏编程

让你比95%的人更懂Pythonic的内置模块:collections

Python的集合(collections)模块,为很多用其他方法很难实现的场景提供了解决方案。 本文我们将会学习该模块的抽象概念是如何产生的,日后处理不同问题...

1795
来自专栏灯塔大数据

技术 | Python从零开始系列连载(十一)

导读 为了解答大家初学Python时遇到各种常见问题,小灯塔特地整理了一系列从零开始的入门到熟练的系列连载,每周五准时推出,欢迎大家学积极习转载~ 上一期学习了...

34610
来自专栏C/C++基础

C++inline函数简介

inline函数是由inline关键字来定义,引入inline函数的主要原因是用它替代C中复杂易错不易维护的宏函数。

762
来自专栏Java技术栈

java程序员被误导的一个概念,90%人不知道

我们经常听说List是有序且重复的,Set是无序不重复的。这里有个误区,这里说的顺序有两个概念,一是按添加的顺序排列,二是按自然顺序a-z排列。Set并不是无序...

3176

扫描关注云+社区