UEGrids.js

最近项目需要,接触了一下angularjs, 感觉双向绑定非常的强大,于是花几个小时的时间,把项目里要实现的一个类似表格数据交换的功能写了一下,angular支持module封装,上手后使用感觉思维也很清晰。于是贴一下主要代码:

首先是页面

<uegrids grids="gridsLeft" selected="selected" width="200"></uegrids>

这个是自定义的uegrids指令,在controller里面我定议了一个数组对象$scope.gridsLeft, 一个变量$scope.selected,和一个元素属性 width = 200;

$scope.gridsLeft = [
                    { id: "10010", with: "red", value: 0.2000 },
                    { id: "10010", with: "red", value: 0.2010 },
                    { id: "10010", with: "red", value: 0.2013 },
                    { id: "10010", with: "red", value: 0.2014 },
                    { id: "10010", with: "red", value: 0.2002 },
                    { id: "10010", with: "red", value: 0.2012 },
                    { id: "10010", with: "red", value: 0.2020 },
                    { id: "10010", with: "red", value: 0.2030 },
                    { id: "10010", with: "red", value: 0.2003 },
                    { id: "10010", with: "red", value: 0.2013 },
                    { id: "10010", with: "red", value: 0.2025 },
                    { id: "10010", with: "red", value: 0.2034 },
                    { id: "10010", with: "red", value: 0.2005 },
                    { id: "10010", with: "red", value: 0.2015 },
                    { id: "10010", with: "red", value: 0.2026 },
                    { id: "10010", with: "red", value: 0.2038 }
            ];

        $scope.selected = false;
        $scope.$watch("selected", function(bool) {
          console.log(bool);
        });

 

我使用的是编辑器webstrom 8,代码自动提示完成包含了requirejs、angularjs,真的很方便

然后看下指令部分:

angular.module(‘directives‘, []).
        factory("setPos", function() {
            return function(width, rows) {
                console.time("setPos");
                var posArr = [],
                    liW = width/ 4,
                    liH = 25,
                    row = 0,
                    col = 1;

                for (var row = 0; row < rows; row++) {
                    for (var col = 1; col <= 4; col++) {

                        posArr.push({left:col*liW, top:row*liH});
                    }
                }
                console.timeEnd("setPos");
                return posArr;
            }
        }).
        directive(‘uegrids‘, function() {
            return {
                restrict: "E",
                scope: {
                    grids: "=",
                    selected: "="
                },
                templateUrl: "template/uegrids.html",
                controller: function($scope, $element, setPos) {
                    // console.info($scope.grids, $element);
                    // this.grids = $scope.grids;
                    var width = $element.attr("width") || 200,
                        rows = $scope.grids.length/ 4,
                        selectedId = null;

                    $scope.setGrid = function(grid) {
                        return grid.with ? grid.with : "gray";
                    };

                    $scope.selectGrid = function(event) {
                        if (event.toElement.tagName.toUpperCase() == "LI") {
                            $(event.toElement).parent().find("li").removeClass("active");
                            $(event.toElement).addClass("active");
                            var curId = Number(event.toElement.dataset.id), temp;
                            if (curId != selectedId && selectedId !== null) {
                                $scope.selected = true;
                                // 交换数据
                                temp = $scope.grids[selectedId];
                                $scope.grids[selectedId] = $scope.grids[curId];
                                $scope.grids[curId] = temp;
                                console.log("交互后,索引是", selectedId);
                                $(event.toElement).removeClass("active");
                                selectedId = null;
                            } else {
                                // $scope.selected = false;
                                selectedId = curId;
                            }
                        }
                        event.preventDefault();
                        return false;
                    };

                    var gridPos = setPos(width, rows);
                    $scope.setPosition = function(index) {
                        var pos = gridPos[index];
                        return "left: "+pos.left+"px; top: "+pos.top+"px";
                    };

                },
                link: function(scope, element, attribute, ctrl) {
                    /*console.log(scope.grids, element, attribute);*/
                    var lis = element.find("li");
                },
                replace: true
            }
        })

这里代码有点罗嗦,反正也是第一次写指令,将就一下。然后就是我在requirejs引用了jquery, angualrjs自身有个qlite的dom操作模块,如果检测到引入了jquery,就会优先使用了jquery,我在指令里使用了dom操作。可能这不太符合angualrjs的提倡,但这里只是一个主要思路,并不是一个完整的项目,所以就不按use strict方式来了。

另外发现,在angularjs指令里做循环或计算操作,会使性能下降很快,比如:

<div class="uegrids" ng-click="selectGrid($event)">
    <ul>
        <li ng-class="setGrid(grid)" style="{{setPosition($index)}}" data-id="{{$index}}" ng-repeat="grid in grids">{{grid.value}}</li>
    </ul>
</div>

假如我在li里加一个ng-style="setPosition($index, grid)", 然后我在setPosition里面去获取$index, grid,然后一堆的计算位置,不好意思,在directive的controller或link里,可能会抛异常。比较好的做法是在controller里预先计算好位置,存到一个变量里面,这里setPosition($index), 只返回变量的引用值即可,效率很高。

数据的交互同于angular的双向绑定,让开发者减少了很多dom的操作,只需要通过简单的ng-click, ng-bind就可以在controller改变数据的同时,迅速地反映在页面上, 而且更妙的是使用scope: {a: "="},使得指令里的a和业务层controller里的a关联了起来,同时发生改变。

然后我在业务层提交的数据,也是指令层交换后的数据,一点也不需要额外做数据修改的处理。

时间: 2024-08-24 07:44:56

UEGrids.js的相关文章

【API】高德地图API JS实现获取坐标和回显点标记

1.搜索+选择+获取经纬度和详细地址 2.回显数据并点标记 3.实现 第一步:引入资源文件 <!--引入高德地图JSAPI --><script src="//webapi.amap.com/maps?v=1.3&key=在官网申请一个key"></script><!--引入UI组件库(1.0版本) --><script src="//webapi.amap.com/ui/1.0/main.js">

js跨域

第一次写博客,好紧张,不知道能写成啥样,哈哈哈. 自己的一知片解,有错请多多指教,嘻嘻嘻. 一.何为跨域? 只要协议.域名.端口后任何一个不同,就是跨域. 举个例子: http://www.example.com 协议不同 https://www.example.com http://www.example.com 域名不同 http://www.test.com http://www.example.com 端口不同 http://www.example.com:81 注意:ip相同,域名不同

Vue.js学习笔记:属性绑定 v-bind

v-bind  主要用于属性绑定,Vue官方提供了一个简写方式 :bind,例如: <!-- 完整语法 --> <a v-bind:href="url"></a> <!-- 缩写 --> <a :href="url"></a> 绑定HTML Class 一.对象语法: 我们可以给v-bind:class 一个对象,以动态地切换class.注意:v-bind:class指令可以与普通的class特

node.js的安装及配置

一.安装 直接在浏览器搜索node.js,在官网上下载(一般旧版的更加稳定,比如下载4.4.7版本) 点击DOWNLOADS 往下翻,点击Previous Release Windows下载msi(64位/32位) 根据提示一步步安装,安装之后的文件夹如下: 在cmd命令行下输入node -v,如果出现如下,说明安装成功: 二.关于配置 在安装路径下新建两个文件夹: 创建完两个空文件夹之后,打开cmd命令窗口,输入 npm config set prefix "D:\Program Files

Node.js 使用angularjs取得Nodejs http服务端返回的JSON数组示例

server.js代码: // 内置http模块,提供了http服务器和客户端功能(path模块也是内置模块,而mime是附加模块) var http=require("http"); // 创建服务器,创建HTTP服务器要调用http.createServer()函数,它只有一个参数,是个回调函数,服务器每次收到http请求后都会调用这个回调函数.服务器每收到一条http请求,都会用新的request和response对象触发请求函数. var server=http.createS

Knockout.js简介

Knockout是一款很优秀的JavaScript库,通过应用MVVM模式使JavaScript前端UI简单化.任何时候你的局部UI内容需要自动更新,KO都可以很简单的帮你实现,并且非常易于维护. Knockout的3个核心功能是: ? 属性监控与依赖跟踪 ? 声明式绑定 ? 模板机制 MVVM Model-View-View Model (MVVM)是一种创建用户界面的设计模式. ? Model:用于存储应用程序数据,表示业务领域的对象和数据操作,并且独立于任何界面. 当使用KO的时候,通常是

vue.js 入门

简单一句话来描述:vue.js是一个前端框架. 官方文档:https://cn.vuejs.org/v2/guide/index.html 虽然,我以前也会改一些前端样式,但主要是基于HTML和CSS,HTML主要控制页面的结构,CSS主要来控制样式.涉及到JavaScript就比较小白了. 我花了一个下午照着官方文档做练习,当然是只创建一个xxx_demo.html文件,在<script></script> 标签对之间写 Vue.js语法.这不算错,但在工程化的今天,这么学得猴年

JS高程3:JSON

JSON,JavaScript Object Notation,JS对象表示法,是目前最常见的结构化数据传输形式. JSON并非编程语言,而是一种数据结构,像mp4.avi一样,只是一种数据格式而已.(数据结构可以包含很多数据类型) JSON值的类型 简单值 对象 数组 简单值:字符串.数字.布尔值和null,注意不包括undefined. 注意:JSON中的字符串必须用双引号. 对象:对象就是无序的键值对,而键值中的值也可以是简单值.对象或者数组. 注意:JSON中对象的属性必须用双引号括起来

js装饰器

本文是廖雪峰老师js教程的学习笔记 JavaScript的所有对象都是动态的,即使内置的函数,我们也可以重新指向新的函数. 利用apply(),我们还可以动态改变函数的行为. 现在假定我们想统计一下代码一共调用了多少次parseInt(),可以把所有的调用都找出来,然后手动加上count += 1,不过这样做太傻了.最佳方案是用我们自己的函数替换掉默认的parseInt(): var count = 0; var oldParseInt = parseInt; // 保存原函数 window.p