前端demo组织优化(进阶思路)

最近在看老项目的代码,一个客户端的js代码,几千行的代码,全是function(){} var...的垂直布局,真的是要感动的哭了。哈哈

一开始都是这样,想实现什么功能,不管三七二十一,function走起,最终堆起无数个变量和函数来完成一个画面的js。我也是,但过段时间自己去改代码bug或者加功能的时候,我的天,这是我写的吗,什么时候写的,怎么理不清思路了,而且,修改一个地方其他地方也得改,改完了还容易出新bug,偶尔都会忘了是自己写的

慢慢的代码看多了点,了解了些js的模块封装的一些方式,面向对象的相关思想,越来越觉得易读、易改的代码应该需要更好的组织形式

下面来看看平时写代码,面向过程到面向对象,梳理前端开发思路

首先需要知道你要做什么需求得明白再来分析代码怎么写

功能需求

如下图,功能比较简易:

选择之后添加,展示区便陈列:

展示区点击‘X’的时候去除当前内容,选择区相应也取消对应的勾选:

优化前代码


<!DOCTYPE html>
 <html>
 
 <head>
     <meta charset="UTF-8">
     <title>多选框问题</title>
 </head>
 
 <body>
     <!--<input type="text" data-bind-content="name" />
         <span data-bind-content='name'></span>-->
 
     <h4>选择区</h4>
     <div>
         
         <ul id="ul1">
             <li>全选<input type="checkbox" name="checkall" /></li>
             <li><input type="checkbox" name="checkthis" /><span>1</span></li>
             <li><input type="checkbox" name="checkthis" /><span>2</span></li>
             <li><input type="checkbox" name="checkthis" /><span>3</span></li>
             <li><input type="checkbox" name="checkthis" /><span>4</span></li>
             <li><input type="checkbox" name="checkthis" /><span>5</span></li>
         </ul>
     </div>
     
     <button id="add">添加</button>
     <h4>展示区</h4>
     <ul id="ul2"></ul>
 </body>
 <script src="http://libs.baidu.com/jquery/2.0.0/jquery.min.js"></script>
 <script type="text/javascript">
     //封装
     var checkBox = (function () {
         var globalV = [];
         var yourChose = function (tableId, addClickId, showId) {
             console.log($('#' + tableId + ' input[name=checkall]'));
             //全选
             $('#' + tableId + ' input[name=checkall]').click(function () {
                 //如果选择全选, 所有的选择框都选中,去除全选,所有的选择框去除选中    
                 if ($(this).prop('checked')) {
                     $('#' + tableId + ' input[name=checkthis]').prop('checked', true);
                     //全选的时候,将所有选框的数据取出来传给全局变量globalV
                     $('#' + tableId + ' input[name=checkthis]').each(function (i, ele) {
                         var choseDate = {};
                         choseDate.isChecked = true;
                         choseDate.id = $(ele).parent().children('span').html();
                         globalV.push(choseDate);
                     });
                 } else {
                     $('#' + tableId + ' input[name=checkthis]').prop('checked', false);
                 }
                 console.log(globalV);
             })
             //对各个选择框绑定事件
             $('#' + tableId).on('change', 'input[name=checkthis]', function () {
                 var arr = [];//存储每个选择框的状态
                 var choseDate = {};//存储被选中的选择框的数据
                 //<li><input type="checkbox" name="check-this" /><span>3</span></li>获取span里面的值
                 var this_value = $(this).parent().children('span').html();
                 //遍历每个选择框取选择的状态
                 $('#' + tableId + ' input[name=checkthis]').each(function (i, ele) {
                     arr.push($(ele).prop('checked'));
                 });
                 //如果有未选中的状态,去除全选框的选中状态,否则保留添加全选框的的选中状态
                 if (arr.indexOf(false) == -1) {
                     $('#' + tableId + ' input[name=checkall]').prop('checked', true);
                 } else {
                     $('#' + tableId + ' input[name=checkall]').prop('checked', false);
                 }
                 //对应每个选择框的change事件,如果这个选择框选中,则存储这个选择框的数据,否则遍历存储数据的变量,移除这个取消选中的的选择框的数据
                 if ($(this).is(':checked')) {
                     choseDate.isChecked = true;
                     choseDate.id = this_value;
                     globalV.push(choseDate);
                 } else {
                     for (var i = 0; i < globalV.length; i++) {
                         if (this_value == globalV[i].id) {
                             globalV.splice(i, 1);
                         }
                     }
                 }
                 console.log(globalV);
             });
             //点击添加按钮的事件
             $('#'+addClickId).click(function (e) {
                 e.preventDefault();
                 $('#'+showId).empty();//清空展示区里面的内容
                 console.log(globalV);
                 //如果没有选中任何选择框,则弹出提示
                 if (globalV.length == 0) {
                     alert('请先选择!');
                 } else {
                     //如果选中了一些选择框,则全局变量数据不为空,开始遍历全局变量
                     for (var j = 0; j < globalV.length; j++) {
                         //按照全局变量globalV,给展示区创建元素;(包含了删除按钮)
                         var liElement = '<li>\
                                             <span>'+ globalV[j].id + '</span>\
                                             <p style="display:inline-block;width:20px;height:20px;background-color:red;border-radius:50%;text-align:center">X</p>\
                                         </li>';
                         $('#'+showId).append(liElement);
                     }
                     //给删除按钮添加点击事件
                     $('#'+showId).on('click', 'p', function () {
                         //var findAndChangeState=$(this).parent('li').children('span').html();
                         //找到这个删除按钮对应的父级标签li下面的span标签的内容;注意:这个是简化;就放在了标签里面,实际情况可能是个属性,获取的这个值对应一个选择框
                         //由这个值来查找对应的选择框,从而改变选择框的状态;
                         //这里是点击了删除按钮,那么与他对应的选择框的选中状态也会被去除
                         var findAndChangeState = $(this).parent('li').children('span').html();
                         //遍历选择框找到与删除按钮对应的选择框,将其状态改为未选中,同时将全选的选择框也改为未选中
                         $('#' + tableId + ' input[name=checkthis]').each(function (i, ele) {
                             if ($(this).parent().children('span').html() == findAndChangeState) {
                                 $(this).parent().children('input').prop('checked', false);
                                 $('#' + tableId + ' input[name=checkall]').prop('checked', false);
                             }
                         });
                         //改完之后这个删除按钮对应的父级标签
                         $(this).parent('li').remove();
                     })
                 }
             })
         };
         return {
             globalV:globalV,
             yourChose:yourChose
         }
     })()
     checkBox.yourChose('ul1', 'add', 'ul2')
 </script>
 
 </html>

优化后代码

<!DOCTYPE html>
<html>

<head>
    <meta charset="UTF-8">
    <title>多选框问题</title>
</head>

<body>
    <!--<input type="text" data-bind-content="name" />
        <span data-bind-content='name'></span>-->

    <h4>选择区</h4>
    <div>

        <ul id="ul1">
            <li>全选<input type="checkbox" name="checkall" /></li>
            <li><input type="checkbox" name="checkthis" /><span>1</span></li>
            <li><input type="checkbox" name="checkthis" /><span>2</span></li>
            <li><input type="checkbox" name="checkthis" /><span>3</span></li>
            <li><input type="checkbox" name="checkthis" /><span>4</span></li>
            <li><input type="checkbox" name="checkthis" /><span>5</span></li>
        </ul>
    </div>

    <button id="add">添加</button>
    <h4>展示区</h4>
    <ul id="ul2"></ul>
</body>
<script src="http://libs.baidu.com/jquery/2.0.0/jquery.min.js"></script>
<script type="text/javascript">
    //封装
    var checkBox = (function () {

        var globalV = [];
        var yourChose = function (tableId, addClickId, showId) {
            //负责更新数据
            var updateData = function () {
                globalV = [];
                $('#' + tableId + ' input[name=checkthis]').each(function () {
                    if ($(this).is(':checked')) {
                        var choseDate = {};
                        var this_value = $(this).parent().children('span').html();
                        choseDate.isChecked = true;
                        choseDate.id = this_value;
                        globalV.push(choseDate);
                    }
                });

            }

            //负责更新画面
            //checkBox状态
            function fun1() {
                if ($(this).attr("name") == "checkthis") {
                    var arr = [];//存储每个选择框的状态
                    var choseDate = {};//存储被选中的选择框的数据
                    //<li><input type="checkbox" name="check-this" /><span>3</span></li>获取span里面的值
                    var this_value = $(this).parent().children('span').html();
                    //遍历每个选择框取选择的状态
                    $('#' + tableId + ' input[name=checkthis]').each(function (i, ele) {
                        arr.push($(ele).prop('checked'));
                    });
                    //如果有未选中的状态,去除全选框的选中状态,否则保留添加全选框的的选中状态
                    if (arr.indexOf(false) == -1) {
                        $('#' + tableId + ' input[name=checkall]').prop('checked', true);
                    } else {
                        $('#' + tableId + ' input[name=checkall]').prop('checked', false);
                    }
                } else {
                    //如果选择全选, 所有的选择框都选中,去除全选,所有的选择框去除选中
                    if ($(this).prop('checked')) {
                        $('#' + tableId + ' input[name=checkthis]').prop('checked', true);
                    } else {
                        $('#' + tableId + ' input[name=checkthis]').prop('checked', false);
                    }
                }
            }
            //展示区状态(新增)
            function fun2() {
                $('#' + showId).empty();//清空展示区里面的内容
                updateData();
                //如果没有选中任何选择框,则弹出提示
                if (globalV.length == 0) {
                    alert('请先选择!');
                } else {
                    //如果选中了一些选择框,则全局变量数据不为空,开始遍历全局变量
                    for (var j = 0; j < globalV.length; j++) {
                        //按照全局变量globalV,给展示区创建元素;(包含了删除按钮)
                        var liElement = '<li>\
                                             <span>'+ globalV[j].id + '</span>\
                                             <p style="display:inline-block;width:20px;height:20px;background-color:red;border-radius:50%;text-align:center">X</p>\
                                         </li>';
                        $('#' + showId).append(liElement);
                    }
                    //给删除按钮添加点击事件
                    bindEvent('#' + showId + ' p', "click", event.removeLi);
                }
            }
            //展示区状态(删除)
            function fun3() {
                //var findAndChangeState=$(this).parent('li').children('span').html();
                //找到这个删除按钮对应的父级标签li下面的span标签的内容;注意:这个是简化;就放在了标签里面,实际情况可能是个属性,获取的这个值对应一个选择框
                //由这个值来查找对应的选择框,从而改变选择框的状态;
                //这里是点击了删除按钮,那么与他对应的选择框的选中状态也会被去除
                var findAndChangeState = $(this).parent('li').children('span').html();
                //遍历选择框找到与删除按钮对应的选择框,将其状态改为未选中,同时将全选的选择框也改为未选中
                $('#' + tableId + ' input[name=checkthis]').each(function (i, ele) {
                    if ($(this).parent().children('span').html() == findAndChangeState) {
                        $(this).parent().children('input').prop('checked', false);
                        $('#' + tableId + ' input[name=checkall]').prop('checked', false);
                    }
                });
                //改完之后这个删除按钮对应的父级标签
                $(this).parent('li').remove();
            }

            //负责注册事件
            var event = {
                select: fun1,
                add: fun2,
                removeLi: fun3
            };
            var bindEvent = function (selector, type, fun) {
                $(selector).bind(type, fun);
            };
            //对各个选择框绑定事件
            bindEvent('#' + tableId + ' input[type=checkbox]', "click", event.select);
            //点击添加按钮的事件
            bindEvent('#' + addClickId, "click", event.add);
        };

        return {
            globalV: globalV,
            yourChose: yourChose
        }
    })()
    checkBox.yourChose('ul1', 'add', 'ul2');
</script>

</html>

更改后的版本里的代码其实都是原来的代码,但组织后的效果是:事件统一绑定(bindEvent),画面统一更新(fun1、fun2、fun3),数据统一设定(updateData)。

区分的很清楚,哪儿出错找哪儿,几乎不会交叉。而且比较容易拓展,像事件可以继续bindEvent绑定,画面更新的函数可以相应与fun1、fun2、fun3并列添加,数据的额外处理可以添加到updateData里。

这仅仅是代码组织上的优化,其实代码本身也有很多可以改进的地方,像全选的判定、选择区联动删除等都有更好的思路和代码实现。

发现,其实这里面已经有mvc的影子了,各司其职,分工明确,事件绑定那部分就算是一个弱controller,绑定事件,分发事件响应函数;更新画面状态部分相当于view了,更新画面;updateData更新数据部分更新的就是modle;

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,016评论 25 707
  • 在线阅读 http://interview.poetries.top[http://interview.poetr...
    程序员poetry阅读 114,344评论 24 450
  • 姓名:岳沁 学号:17101223458 转载自:http://www.cnblogs.com/fsyz/p/80...
    丘之心阅读 481评论 0 0
  • 用rails的都用过where方法,总结了一些基本的,欢迎提供更多的内容。 where方法用来指定限制获取记录的条...
    GALAXY_ZMY阅读 4,978评论 0 3
  • 原创160篇 文 / 烽火 走进“南京大屠杀”陈列馆 七月的暑天骤然下起了雪…… 历史在揭过残暴的一幕时 匆匆地画...
    慧心育子阅读 228评论 0 1