13、angular1组件之pass-word密码、single-select|both-select消长、on-off开关、simple-dialog多层弹窗(含插槽ng-transclude)、3种search-select搜索下拉、input-select树结构 、dir-tree树结构(3150行)
一、pass-word组件(自定义密码输入框,可预览) 使用场景:修改密码时,对密码进行多方位验证,并显示验证结果。 <!DOCTYPE html> <html lang="en" ng-app="myModel"> <head> <meta charset="UTF-8"> <title>passWord</title> <script type="text/javascript" src="https://cdn.bootcss.com/angular.js/1.6.2/angular.js"></script> </head> <body> <div ng-controller="thisCtrl"> <pass-word label-width="150" is-must-fill="true" label-text="原密码" this-type="password" placeholder="原密码" is-disabled="false" self-show='pastShow' is-past-input='true' self-data="oldPassword" verify="verify" ></pass-word> <pass-word label-width="150" is-must-fill="true" label-text="输入新密码" this-type="password" placeholder="输入新密码" is-disabled="false" is-first-input='true' key-of-reg-exp="simplePassword" self-data="params.newPassword" sibling-data="newPasswordSecond" one-tip="oneTip" many-tips="manyTips" self-show='selfShow' sibling-show='siblingShow' ></pass-word> <pass-word label-width="150" is-must-fill="true" label-text="再次输入新密码" this-type="password" placeholder="再次输入新密码" is-disabled="false" key-of-reg-exp="simplePassword" self-data="newPasswordSecond" sibling-data="params.newPassword" one-tip="oneTip" many-tips="manyTips" self-show='siblingShow' sibling-show='selfShow' ></pass-word> </div> </body> </html> <script> var app = angular.module('myModel', []); app.controller('thisCtrl', function ($scope) { $scope.params = {};//发给后台 $scope.pastShow = { isShowOneTip:false, isGreenOneTip:false, isShowManyTips:false, }; $scope.selfShow = { isShowOneTip:false, isGreenOneTip:false, isShowManyTips:false, }; $scope.siblingShow = { isShowOneTip:false, isGreenOneTip:false, isShowManyTips:false, }; $scope.oneTip = {//由当前页传至本组件 success: '自定义_输入符合要求!', empty: '自定义_该项不能为空!', different: '自定义_两次密码输入不一致!', formatError: '自定义_输入不符合要求,请输入以字母开头的4-24个字符!' }; $scope.manyTips= [ '密码要求:', '1.必须包含:数字', '2.必须包含:字母', '3.必须包含:小写字母', '4.必须包含:大写字母', '5.必须包含:下划线', ]; $scope.oldPassword= ''; $scope.params.newPassword=''; $scope.newPasswordSecond= ''; $scope.verify = function() { //把原密码告诉后台,根据后台判断结果,决定是否继续验证 return true } }); app.directive('passWord', function () { var html = ` <div style="display:flex;align-items:center"> <div ng-style="{width:(labelWidth||130)+'px'}" style="text-align:right;margin-right:10px;"> <span ng-show="isMustFill" style="color:red;">*</span> <label ng-bind="labelText||'密码'"></label> </div> <div style="padding:10px 0; position:relative;"> <input type="{{ thisType ? thisType : 'text' }}" ng-model="selfData" placeholder="{{placeholder||''}}" ng-disabled="isDisabled" ng-focus="focus()" ng-change="changeOrBlur()" ng-blur="changeOrBlur()" ng-trim="false" style="width:290px;height:30px" > <img src="{{servicePicture.eyeClose}}" ng-click="isShowEyeOpen=true;thisType='text'" ng-show="!isShowEyeOpen" style="width:14px;height:14px;position:absolute;right:10px;top:20px;cursor:pointer;padding:5px;" > <img src="{{servicePicture.eyeOpen}}" ng-click="isShowEyeOpen=false;thisType='password'" ng-show="isShowEyeOpen" style="width:14px;height:14px;position:absolute;right:10px;top:20px;cursor:pointer;padding:5px;" > </div> <div style="position:relative"> <div ng-show="selfShow.isShowOneTip"> <img src="{{selfShow.isGreenOneTip ? servicePicture.correct : servicePicture.error}}" style="width:14px;height:14px;position:relative;left:5px;"> <span ng-bind="singleTipText" ng-style="selfShow.isGreenOneTip?{ \'color\':\'green\'}:{\'color\':\'red\'}" style="position:relative;left:5px;"> ></span> </div> <div ng-show="selfShow.isShowManyTips && manyTips.length>0" style="position:absolute;top:0;display:flex"> <img src="{{servicePicture.triangle}}" style="width:14px;height:14px;position:relative;z-index:3;left:5px;" > <div style="width:200px;border:1px solid #DCDCDC;position:relative;top:-5px;left:3px;padding:5px;"> <div ng-repeat="li in manyTips" ng-bind="li"></div> </div> </div> </div> </div> `; return { restrict: 'E', template: html, scope: { //以下是16种配置项。这是输入框组件,适用于密码输入、普通输入(isPastInput/isFirstInput:true)、旁边提示和弹窗提示。 labelWidth: '@labelWidth',//标签宽度 labelText: '@labelText',//标签文字 placeholder: '@placeholder',//输入框内提示文字 thisType: '@thisType',//输入框类型(文字或密码) keyOfRegExp: '@keyOfRegExp',//会用到的正则 isMustFill: '=isMustFill',//该项是否必填 isDisabled: '=isDisabled',//该项是否禁用 isPastInput: '=isPastInput',//该项是不是原密码xxxxxxxxxxx isFirstInput: '=isFirstInput',//该项是不是第一次输入新密码xxxxxxxxxxxx selfData: '=selfData',//自身数据============ siblingData: '=siblingData',//兄弟数据=============== oneTip: '=oneTip',//单条提示 manyTips: '=manyTips',//多条提示 selfShow: '=selfShow',//自身的提示是否显示出来============ siblingShow: '=siblingShow',//兄弟的提示是否显示出来============= verify: '=verify',//verify是函数,但按照变量传进来更好xxxxxxxxxxx }, replace: true, controller: function ($scope, serviceRegExp, servicePicture) { $scope.servicePicture = servicePicture; $scope.inputChange = function (params, message) { var params = params; var config = { init: [false, false, false], focus: [false, false, true], empty: [true, false, false], formatError: [true, false, false], different: [true, false, false], wrong: [true, false, false], success: [true, true, false], }; $scope.selfShow.isShowOneTip = config[params][0]; $scope.selfShow.isGreenOneTip = config[params][1]; $scope.selfShow.isShowManyTips = config[params][2]; $scope.singleTipText = ($scope.oneTip && $scope.oneTip[params]) || message; }; $scope.focus = function () { if($scope.isPastInput){ $scope.inputChange('init'); return } if($scope.isFirstInput){ $scope.siblingData=''; $scope.siblingShow = { isShowOneTip:false, isGreenOneTip:false, isShowManyTips:false, }; } $scope.inputChange('focus'); }; $scope.changeOrBlur = function () { if ($scope.isMustFill && $scope.selfData=="") {//判断该项是否可以为空 $scope.inputChange('empty', '该项不能为空!'); return } if ($scope.keyOfRegExp && !(serviceRegExp[$scope.keyOfRegExp].test($scope.selfData))) { $scope.inputChange('formatError', '格式错误!'); return } if (!$scope.isPastInput && !$scope.isFirstInput && $scope.selfData != $scope.siblingData) { $scope.inputChange('different', '两次密码输入不一致!'); return } if(angular.isFunction($scope.verify) && !$scope.verify()){//验证原密码是否正确 $scope.inputChange('wrong', '原密码输入错误!'); return }; $scope.inputChange('success', '输入符合要求!'); }; }, link: function (scope, ele, attrs) { } }; }); app.factory('serviceRegExp', function () { return { simplePassword: /^[a-zA-Z][a-zA-Z0-9]{4,24}$/,//简单密码验证,以大小写字母开头,由大小写字母和数字构成的,共5-25位的密码 complexPassword: /^(?![a-zA-z]+$)(?!\d+$)(?![!@#$%^&*?_|~=+\-<>[{}/\'\":;,.\]*]+$)(?![a-zA-z\d]+$)(?![a-zA-z!@#$%^&*?_|~=+\-<>[{}/\'\":;,.\]*]+$)(?![\d!@#$%^&*?_|~=+\-<>[{}/\'\":;,.\]*]+$)[a-zA-Z\d!@#$%^&*?_|~=+\-<>[{}/\'\":;,.\]*]{6,25}$/,//复杂密码验证,同时包含字母、数字、特殊符号如!@#$%^&*?_|~=+\-<>[{}/\'\":;,.\] ipFrom0To255: /^(1\d{2}|2[0-4]\d|25[0-5]|[1-9]\d|\d)\.(1\d{2}|2[0-4]\d|25[0-5]|[1-9]\d|\d)\.(1\d{2}|2[0-4]\d|25[0-5]|[1-9]\d|\d)\.(1\d{2}|2[0-4]\d|25[0-5]|[1-9]\d|\d)$/,//ip地址验证,从0.0.0.0~255.255.255.255 portFrom0To65535:/^([0-9]|[1-9]\d{1,3}|[1-5]\d{4}|6[0-4]\d{3}|65[0-4]\d{2}|655[0-2]\d|6553[0-5])$/, mac:/^[A-Fa-f\d]{2}:[A-Fa-f\d]{2}:[A-Fa-f\d]{2}:[A-Fa-f\d]{2}:[A-Fa-f\d]{2}:[A-Fa-f\d]{2}$/, mobilePhone: /^1[3456789]\d{9}$/,//手机号码验证,中国大陆 numberNotLimit: /^(0|[1-9][0-9]*)$/,//0或非0开头的数字 numberWithLimit: /^(0|[1-9][0-9]{3,4})$/,//0或非0开头的数字4-5位 emailNotLimit: /^\w+([-+.]\w+)*@\w+([-.]\w+)*\.\w+([-.]\w+)*$/,//邮箱验证 emailWithLimit: /^[A-Za-z\d]+([-_.][A-Za-z\d]+)*@([A-Za-z\d]+[-.])+[A-Za-z\d]{2,4}$/,//邮箱验证 chineseCharacters: /^[\u4e00-\u9fa5]{0,}$/,//汉字验证 blankSpace: /^\s*$/ //零到多个空格 }; }); app.factory('servicePicture', function () { return { eyeClose: '', eyeOpen: '', correct: '', error: '', triangle: '' }; }); </script> 二、single-select和both-select <!DOCTYPE html> <html lang="en" ng-app="myModel"> <head> <meta charset="UTF-8"> <title>onlySelect</title> <script type="text/javascript" src="https://cdn.bootcss.com/angular.js/1.6.2/angular.js"></script> </head> <body> <div ng-controller="thisCtrl"> <div>一、此消彼长式</div> <div>使用场景:</div> <div>有上下两行数据,上行有很多备选项</div> <div>每点击上行的某项,则该项“会”消失并在下行出现</div> <div>每点击下行的某项,则该项“会”消失并在上行出现</div> <div style="margin-bottom:20px;"> <span ng-click="getDataUp()" style="border-radius:5px;padding:2px;background:gray;">点击获取选中数据</span> <span ng-bind="selectedDatasUp"></span> </div> <div style="width:1000px;"> <single-select form-datas-up="formDatasUp" current-datas-up="currentDatasUp" two-label-up="twoLabelUp"></single-select> </div> <div style="width:1000px;border:1px solid gray;margin:100px 0"></div> <div>二、此静彼长式</div> <div>使用场景:</div> <div>有上下两行数据,上行有很多备选项</div> <div>每点击上行的某项,则该项“不”消失并在下行出现,重复点击无效</div> <div>每点击下行的某项,则该项“会”消失,但对上行无影响</div> <div style="margin-bottom:20px;"> <span ng-click="getDataDown()" style="border-radius:5px;padding:2px;background:gray;">点击获取选中数据</span> <span ng-bind="selectedDatasDown"></span> </div> <div style="width:1000px;"> <both-select form-datas-down="formDatasDown" current-datas-down="currentDatasDown" two-label-down="twoLabelDown"></both-select> </div> </div> </body> </html> <script> var app = angular.module('myModel', []); app.controller('thisCtrl', function ($scope) { $scope.formDatasUp = [ { value: '0', label: '选项一' }, { value: '1', label: '选项二' }, { value: '2', label: '选项三' }, { value: '3', label: '选项四' }, { value: '4', label: '选项五' } ]; $scope.currentDatasUp = []; $scope.twoLabelUp = ['上面标签:','下面标签:']; $scope.getDataUp = function(){ $scope.selectedDatasUp=angular.copy($scope.currentDatasUp); }; //////////////////////////////////////////////////////////////////////////////////////// $scope.formDatasDown = [ { value: '0', label: '选项一' }, { value: '1', label: '选项二' }, { value: '2', label: '选项三' }, { value: '3', label: '选项四' }, { value: '4', label: '选项五' } ]; $scope.currentDatasDown = []; $scope.twoLabelDown = ['备选项:','被选项:']; $scope.getDataDown = function(){ $scope.selectedDatasDown=angular.copy($scope.currentDatasDown); }; }); app.directive('singleSelect', function () { var html = ` <div> <div style="width:100%;height:30px;line-height:30px;border:1px solid green;padding:5px;"> <span ng-bind="twoLabelUp[0]"></span> <span ng-repeat="upData in upDatas track by $index" ng-bind="upData.label" ng-click="clickUpUp(upData.value)" style="border-radius:5px;display:inline-block;margin-right:10px;padding:2px;background:gray"></span> </div> <div style="width:100%;height:30px;line-height:30px;border:1px solid red;padding:5px;"> <span ng-bind="twoLabelUp[1]"></span> <span ng-repeat="downData in downDatas track by $index" ng-bind="downData.label" ng-click="clickUPDown(downData.value)" style="border-radius:5px;display:inline-block;margin-right:10px;padding:2px;background:gray"></span> </div> </div> `; return { restrict: 'E', template: html, scope: { formDatasUp: '=formDatasUp', currentDatasUp: '=currentDatasUp', twoLabelUp: '=twoLabelUp', }, replace: true, controller: function ($scope) { $scope.upDatas = angular.copy($scope.formDatasUp); $scope.downDatas = []; $scope.clickUpUp = function (valueOuter) { angular.forEach($scope.upDatas,function(value,index){ if(valueOuter===value.value){ $scope.upDatas.splice(index,1); $scope.downDatas.push(value); $scope.currentDatasUp.push(value.value); console.log($scope.currentDatasUp) } }) }; $scope.clickUPDown = function (valueOuter) { angular.forEach($scope.downDatas,function(value,index){ if(valueOuter===value.value){ $scope.upDatas.push(value); $scope.downDatas.splice(index,1); var thisIndex=$scope.currentDatasUp.indexOf(value.value) if(thisIndex>-1){ $scope.currentDatasUp.splice(thisIndex,1); } console.log($scope.currentDatasUp) } }) }; }, link: function (scope, ele, attrs) { } }; }); app.directive('bothSelect', function () { var html = ` <div> <div style="width:100%;height:30px;line-height:30px;border:1px solid green;padding:5px;"> <span ng-bind="twoLabelDown[0]"></span> <span ng-repeat="upData in upDatasDown track by $index" ng-bind="upData.label" ng-click="clickDownUp(upData)" style="border-radius:5px;display:inline-block;margin-right:10px;padding:2px;background:gray"></span> </div> <div style="width:100%;height:30px;line-height:30px;border:1px solid red;padding:5px;"> <span ng-bind="twoLabelDown[1]"></span> <span ng-repeat="downData in downDatasDown track by $index" ng-bind="downData.label" ng-click="clickDownDown(downData,$index)" style="border-radius:5px;display:inline-block;margin-right:10px;padding:2px;background:gray"></span> </div> </div> `; return { restrict: 'E', template: html, scope: { formDatasDown: '=formDatasDown', currentDatasDown: '=currentDatasDown', twoLabelDown: '=twoLabelDown', }, replace: true, controller: function ($scope) { $scope.upDatasDown = angular.copy($scope.formDatasDown); $scope.downDatasDown = []; $scope.clickDownUp = function (data) { var thisIndex=$scope.currentDatasDown.indexOf(data.value) if(thisIndex>-1){ return }else{ $scope.downDatasDown.push(data); $scope.currentDatasDown.push(data.value); } }; $scope.clickDownDown = function (data,index) { $scope.downDatasDown.splice(index,1); var thisIndex=$scope.currentDatasDown.indexOf(data.value) if(thisIndex>-1){ $scope.currentDatasDown.splice(thisIndex,1); } }; }, link: function (scope, ele, attrs) { } }; }) </script> 三、on-off组件(真实逻辑,点击后,转圈、向后台发送请求、返回数据、给开关赋值、开关展示状态。点击后,立即改成相反状态,是不对的。) <html lang="en" ng-app="myModel"> <head> <meta charset="UTF-8"> <title></title> <script type="text/javascript" src="https:cdn.bootcss.com/angular.js/1.6.2/angular.js"></script> <style> table{ border-collapse: collapse; width:1000px; } table td,table th { padding: 5px; border: 1px solid #cbcbcb; } table thead { background-color: #e0e0e0; color: #000; text-align: left; } </style> </head> <body> <div ng-controller="thisCtrl"> <on-off this-switch="parentSwitch"> <span>{{parentSwitch.state?"总开关已开启":"总开关已关闭"}}。总开关关闭时,不能操作分开关!</span> </on-off> <table> <thead> <tr> <th>序号</th> <th>数据1</th> <th>数据2</th> <th>数据3</th> <th>开关1</th> <th>开关2</th> </tr> </thead> <tbody> <tr ng-repeat="item in tableDatas track by $index"> <td ng-bind="$index+1"></td> <td ng-bind="item.value1"></td> <td ng-bind="item.value2"></td> <td ng-bind="item.value3"></td> <td> <on-off parent-switch="parentSwitch" this-switch="item.value4"></on-off> </td> <td> <on-off parent-switch="parentSwitch" this-switch="item.value5"></on-off> </td> </tr> </tbody> </table> </div> </body> </html> <script> var app = angular.module('myModel', []); app.controller('thisCtrl', function ($scope) { //以下通过向后台请求,获取总开关的数据fromServerParent,经加工后为$scope.parentSwitch var fromServerParent={ isOn:true }; $scope.parentSwitch={ state:fromServerParent.isOn, callback:function(){ this.state=!this.state; } }; //以下通过向后台请求,获取分开关的数据fromServerData,经加工后为$scope.tableDatas var fromServerData=[ {value1:"数据1",value2:"数据2",value3:"数据3",isOn:true,isSwitch:true}, {value1:"数据1",value2:"数据2",value3:"数据3",isOn:true,isSwitch:true}, {value1:"数据1",value2:"数据2",value3:"数据3",isOn:true,isSwitch:true}, {value1:"数据1",value2:"数据2",value3:"数据3",isOn:true,isSwitch:true}, {value1:"数据1",value2:"数据2",value3:"数据3",isOn:true,isSwitch:true}, {value1:"数据1",value2:"数据2",value3:"数据3",isOn:true,isSwitch:true}, {value1:"数据1",value2:"数据2",value3:"数据3",isOn:true,isSwitch:true} ]; var tableDatas=[]; angular.forEach(fromServerData,function(value,key){ var item={}; item.value1=value.value1; item.value2=value.value2; item.value3=value.value3; item.value4={ state:value.isOn, callback:function(){ this.state=!this.state; } }; item.value5={ state:value.isSwitch, callback:function(){ this.state=!this.state; } }; tableDatas.push(item) }) $scope.tableDatas=tableDatas; }); app.directive('onOff', function () {//(含插槽) var html = ` <div style="display:flex;align-content:center;padding-bottom:10px;"> <img ng-src="{{thisSwitch.state?onOff.yes:onOff.no}}" width="65px" height="30px" ng-click="clickSwitch()"/> <span ng-transclude style="height:30px;line-height:30px;padding-left:10px;"></span> </div> `; return { restrict: 'E', template: html, scope: { thisSwitch: '=thisSwitch', parentSwitch: '=parentSwitch', }, transclude:true, replace: false, controller: function ($scope,onOff) { $scope.onOff = onOff; $scope.clickSwitch = function(){ //以下总开关关闭时,不能操作分开关! if($scope.parentSwitch && !$scope.parentSwitch.state){ return '弹窗:父级开关阻止本级开关改变' } if($scope.thisSwitch.isForbid ){ return '弹窗:本级数据的其它条件阻止本级开关改变' } $scope.thisSwitch.callback() }; }, link: function (scope, ele, attr) { } }; }); app.factory('onOff', function () { return { yes: '', no: '', }; }) </script> 四、simple-dialog 1、不可拖拽(含插槽) <!DOCTYPE html> <html lang="en" ng-app="myModel"> <head> <meta charset="UTF-8"> <title>三层不可拖拽弹窗之angular1.6.2版</title> <script type="text/javascript" src="https://cdn.bootcss.com/angular.js/1.6.2/angular.js"></script> <style> .simpleDialog { position: fixed; width: 100%; height: 100%; top: 0; left: 0; display: flex; justify-content: center; align-items: center; } .simpleDialog .mask { position: fixed; width: 100%; height: 100%; top: 0; left: 0; background: black; opacity: 0.5; } .simpleDialog .content { position: fixed; background: white; opacity: 1; display: flex; flex-direction: column; } .simpleDialog .content .title { display: flex; background: blue; color: white; padding: 10px; cursor: pointer; user-select: none; } .simpleDialog .content .conform { display: flex; justify-content: center; padding: 10px; background: blue; } </style> </head> <body> <div ng-controller="thisCtrl"> <button ng-click="clickButton()" style="margin-top: 30px; z-index: 250"> 点击-出现-弹窗 </button> <simple-dialog required-data="requiredDataOut"> 1111111 <simple-dialog required-data="requiredDataMid"> 2222222 <simple-dialog required-data="requiredDataIn"> 33333333 </simple-dialog> </simple-dialog> </simple-dialog> </div> </body> <script> var app = angular.module('myModel', []); app.controller('thisCtrl', function ($scope) { $scope.clickButton = function () { $scope.requiredDataOut.isShow = true; }; $scope.requiredDataOut = { isShow : false, width : '900px', height : '600px', openFn : function () { $scope.requiredDataMid.isShow = true; } }; $scope.requiredDataMid = { isShow : false, width : '600px', height : '400px', openFn : function () { $scope.requiredDataIn.isShow = true; }, }; $scope.requiredDataIn = { isShow : false, width : '300px', height : '200px', }; // 一层弹窗可以如下使用 // $scope.requiredDataOut = { // isShow : false, // }; // $scope.clickButton = function () { // $scope.requiredDataOut.isShow = true; // }; }); app.directive('simpleDialog', function () { var html = ` <div class="simpleDialog" ng-show="requiredData.isShow"> <div class="mask" ng-show="requiredData.isShow"></div> <div class="content" ng-show="requiredData.isShow"> <div class="title"> <span>系统消息</span> </div> <div ng-transclude ng-style="{width:requiredData.width||'800px',height:requiredData.height||'400px'}"></div> <div class="conform"> <button ng-click="close()">关闭</button> <button ng-click="open()">打开</button> </div> <div> </div> `; return { restrict: 'E', template: html, transclude: true, scope: { requiredData: '=' }, controller: function ($scope) { $scope.close = function () { $scope.requiredData.isShow = false; if($scope.requiredData.closeFn)$scope.requiredData.closeFn(); }; $scope.open = function () { if($scope.requiredData.openFn)$scope.requiredData.openFn(); }; } }; }); </script> </html> 2、可拖拽(含插槽) <!DOCTYPE html> <html lang="en" ng-app="myModel"> <head> <meta charset="UTF-8"> <title>三层可拖拽弹窗之angular1.6.2版</title> <script type="text/javascript" src="https://cdn.bootcss.com/angular.js/1.6.2/angular.js"></script> <style> .simpleDialog { position: fixed; width: 100%; height: 100%; top: 0; left: 0; display: flex; justify-content: center; align-items: center; } .simpleDialog .mask { position: fixed; width: 100%; height: 100%; top: 0; left: 0; background: black; opacity: 0.5; } .simpleDialog .content { position: fixed; background: white; opacity: 1; display: flex; flex-direction: column; } .simpleDialog .content .title { display: flex; background: blue; color: white; padding: 10px; cursor: pointer; user-select: none; } .simpleDialog .content .conform { display: flex; justify-content: center; padding: 10px; background: blue; } </style> </head> <body> <div ng-controller="thisCtrl"> <button ng-click="clickButton()" style="margin-top: 30px; z-index: 250"> 点击-出现-弹窗 </button> <simple-dialog required-data="requiredDataOut"> 1111111 <simple-dialog required-data="requiredDataMid"> 2222222 <simple-dialog required-data="requiredDataIn"> </simple-dialog> </simple-dialog> </simple-dialog> </div> </body> <script> var app = angular.module('myModel', []); app.controller('thisCtrl', function ($scope, drag) { $scope.clickButton = function () { $scope.requiredDataOut.isShow = true; drag($scope.requiredDataOut.titleId,$scope.requiredDataOut.contentId); }; $scope.requiredDataOut = { isShow : false, width : '900px', height : '600px', titleId : "titleId1", contentId : "contentId1", openFn : function () { $scope.requiredDataMid.isShow = true; drag($scope.requiredDataMid.titleId,$scope.requiredDataMid.contentId); } }; $scope.requiredDataMid = { isShow : false, width : '600px', height : '400px', titleId : "titleId2", contentId : "contentId2", openFn : function () { $scope.requiredDataIn.isShow = true; drag($scope.requiredDataIn.titleId,$scope.requiredDataIn.contentId); }, }; $scope.requiredDataIn = { isShow : false, width : '300px', height : '200px', titleId : "titleId3", contentId : "contentId3", }; // 一层弹窗可以如下使用 // $scope.requiredDataOut = { // isShow : false, // }; // $scope.clickButton = function () { // $scope.requiredDataOut.isShow = true, // drag(); // }; }); app.directive('simpleDialog', function () { var html = ` <div class="simpleDialog" ng-show="requiredData.isShow"> <div class="mask" ng-show="requiredData.isShow"></div> <div class="content" ng-show="requiredData.isShow" id="{{requiredData.contentId||'contentId'}}"> <div class="title" id="{{requiredData.titleId||'titleId'}}"> <span>系统消息</span> </div> <div ng-transclude ng-style="{width:requiredData.width||'800px',height:requiredData.height||'400px'}">后备内容</div> <div class="conform"> <button ng-click="close()">关闭</button> <button ng-click="open()">打开</button> </div> <div> </div> `; return { restrict: 'E', template: html, transclude: true, scope: { requiredData: '=' }, controller: function ($scope) { $scope.close = function () { $scope.requiredData.isShow = false; angular .element( document.getElementById($scope.requiredData.contentId||'contentId') ) .css({ top: '50%', left: '50%', transform: 'translate(-50%, -50%)' }); if($scope.requiredData.closeFn)$scope.requiredData.closeFn(); }; $scope.open = function () { if($scope.requiredData.openFn)$scope.requiredData.openFn(); }; } }; }); app.factory('drag', function () { return function (wholeTitleId, wholeContentId) { var wholeTitleId = wholeTitleId||'titleId'; var wholeContentId = wholeContentId||'contentId'; var oDiv = document.getElementById(wholeContentId); oDiv.onmousedown = down; function processThis(fn, nowThis) { return function (event) { fn.call(nowThis, event); }; } function down(event) { event = event || window.event; if (event.target.id != wholeTitleId) return; this.initOffsetLeft = this.offsetLeft; this.initOffsetTop = this.offsetTop; this.initClientX = event.clientX; this.initClientY = event.clientY; this.maxOffsetWidth = (document.documentElement.clientWidth || document.body.clientWidth) - this.offsetWidth; this.maxOffsetHeight = (document.documentElement.clientHeight || document.body.clientHeight) - this.offsetHeight; if (this.setCapture) { this.setCapture(); this.onmousemove = processThis(move, this); this.onmouseup = processThis(up, this); } else { document.onmousemove = processThis(move, this); document.onmouseup = processThis(up, this); } } function move(event) { var nowLeft = this.initOffsetLeft + (event.clientX - this.initClientX); var nowTop = this.initOffsetTop + (event.clientY - this.initClientY); //此处可以根据实际需要,限制拖动边界 this.style.left = nowLeft + 'px'; this.style.top = nowTop + 'px'; } function up() { if (this.releaseCapture) { this.releaseCapture(); this.onmousemove = null; this.onmouseup = null; } else { document.onmousemove = null; document.onmouseup = null; } } }; }); </script> </html> 五、3种searchSelect搜索下拉 1、searchSelectFull <!DOCTYPE html> <html lang="en" ng-app="myModel"> <head> <meta charset="UTF-8"> <title>searchSelectFull</title> <script type="text/javascript" src="https://cdn.bootcss.com/angular.js/1.6.2/angular.js"></script> <style> .outer-div{ display: flex; flex-direction: column; flex-wrap: wrap; align-items: top; border: 1px solid #7b8ea5; border-radius: 6px; cursor: pointer; width: 70%; user-select: none; } .select-div{ height: 120px; padding: 5px 0 0 5px; display: flex; justify-content: flex-start; flex-wrap: wrap; overflow-y: auto; } .item-summary{ border: 1px solid #dddddd; padding: 5px 10px; background: #fafafa; margin-right: 5px; margin-bottom: 5px; border-radius: 4px; height:24px; color:#000; } .option-all{ position: absolute; border: 2px solid #dddddd; width:444px; background:#ffffff; z-index: 10; padding:6px } .option-div{ padding: 5px 10px; background: #fafafa; margin-right: 5px; margin-bottom: 5px; border-radius: 4px; text-align: left; } .option-div:hover { background: #cccccc !important; } .yes-selected{ background:#237eff; border-radius:4px; color:#ffffff } .no-selected{ background:#ffffff; border-radius:0; border-bottom:1px solid #e7e7e7; color:#000000 } </style> </head> <body> <div ng-controller="thisCtrl"> <button ng-click="getAllSelectedOptions()">点击获取被选中项</button> <duv>{{result}}</duv> <search-select-full protocol-type="protocolType"></search-select-full> </div> </body> </html> <script> var app = angular.module('myModel', []); app.controller('thisCtrl', function ($scope) { $scope.protocolType = { "yesOptions": [ "c1222", "cnip", "dnp3", "dns", "doip", "egd", "enip", "enip-io", "fins", "ftp", "ftp-data", "ge-srtp", "goose", "hartip", "http", "iec104", "knxip", "mms", "modbus", "opcda", "opcua", "pn-dcp", "pn-mrrt", "pn-ptcp", "pop3", "rdp", "s7", "s7-plus", "sercos", "smb", "smb2", "smtp", "sv", "telnet" ], "allOptions": [ { "server": "c1222", "client": "c1222" }, { "server": "cnip", "client": "cnip" }, { "server": "dnp3", "client": "dnp3" }, { "server": "dns", "client": "dns" }, { "server": "doip", "client": "doip" }, { "server": "egd", "client": "egd" }, { "server": "enip", "client": "enip" }, { "server": "enip-io", "client": "enip-io" }, { "server": "fins", "client": "fins" }, { "server": "ftp", "client": "ftp" }, { "server": "ftp-data", "client": "ftp-data" }, { "server": "ge-srtp", "client": "ge-srtp" }, { "server": "goose", "client": "goose" }, { "server": "hartip", "client": "hartip" }, { "server": "http", "client": "http" }, { "server": "iec104", "client": "iec104" }, { "server": "knxip", "client": "knxip" }, { "server": "mms", "client": "mms" }, { "server": "modbus", "client": "modbus" }, { "server": "opcda", "client": "opcda" }, { "server": "opcua", "client": "opcua" }, { "server": "pn-dcp", "client": "pn-dcp" }, { "server": "pn-mrrt", "client": "pn-mrrt" }, { "server": "pn-ptcp", "client": "pn-ptcp" }, { "server": "pop3", "client": "pop3" }, { "server": "rdp", "client": "rdp" }, { "server": "s7", "client": "s7" }, { "server": "s7-plus", "client": "s7-plus" }, { "server": "sercos", "client": "sercos" }, { "server": "smb", "client": "smb" }, { "server": "smb2", "client": "smb2" }, { "server": "smtp", "client": "smtp" }, { "server": "sv", "client": "sv" }, { "server": "telnet", "client": "telnet" } ], "isFull": true } $scope.getAllSelectedOptions = function(){ var values = $scope.protocolType.getValues(); $scope.result = JSON.stringify(values) }; }); app.directive('searchSelectFull', function () { var html = ` <div class="outer-div" tabindex="-1" ng-blur="getBlurOut()"> <!-- 以下是select --> <div style="display: flex;" ng-click="getOptions($event)"> <div class="select-div"> <div class="item-summary">已选择{{yesOptions.length}}条</div> <div ng-repeat="client in yesOptions track by $index" class="item-summary"> <span ng-bind="client" style="position:relative;top:-7px"></span> <img ng-click="clickFork(yesOptions,client)" class="onlyQC" src="{{checkImg.fork}}" width="24px" height="24px" /> </div> <div class="item-summary" ng-show="yesOptions.length>20">已选择{{yesOptions.length}}条</div> </div> </div> <!-- 以下是option --> <div style="position: relative" ng-show="isShowOptions"> <div class="option-all"> <div> <input type="text" class="form-control" style="width:300px;height:44px;border-radius:4px;margin:4px 0" ng-focus="getFocusInput()" ng-blur="getBlurInput()" ng-change="changeInputValue()" ng-model="selectedOption"/> <span style="padding-bottom: 5px;display:inline-block;width:112px;" ng-style="{'visibility': isShowSelectAll? 'visible':'hidden'}"> <img ng-click="selectAll()" src="{{isSelectAll?checkImg.yes:checkImg.no}}" width="16px" height="16px"/> <span>{{textSelectAll}}</span> </span> </div> <div style="max-height: 360px;overflow: auto;padding-left: 6px;"> <div ng-repeat="option in filterOptions track by $index" class="option-div" ng-click="clickOptions(option)" ng-class="yesOptions.indexOf(option.client)>-1?'yes-selected':'no-selected'"> <span ng-bind="option.client" style="height:34px;line-height:34px;display: inline-block;"></span> </div> </div> </div> </div> </div> `; return { restrict: 'E', template: html, scope: { protocolType: '=protocolType',//所有选项 }, replace: true, controller: function ($scope,$timeout,checkImg) { $scope.checkImg = checkImg; $scope.isSelectAll = true; $scope.isShowSelectAll = true; $scope.isFocusInput = false; $scope.isShowOptions = false; $scope.isClickOptions = false; $scope.isClickSelectAll = false; $scope.yesOptions = $scope.protocolType.yesOptions; $scope.allOptions = $scope.protocolType.allOptions; $scope.filterOptions = angular.copy($scope.allOptions); $scope.textSelectAll = "全部选择"; if($scope.isSelectAll){ $scope.textSelectAll = "已选择"+$scope.yesOptions.length+"条"; } $scope.clickFork = function (yesOptions,client) { angular.forEach(yesOptions,function(option,index){ if(client===option){ yesOptions.splice(index,1); $scope.textSelectAll = "已选择"+$scope.yesOptions.length+"条"; } }) }; $scope.selectAll = function () { $scope.isClickSelectAll = true; $scope.yesOptions = []; $scope.isSelectAll = !$scope.isSelectAll; if($scope.isSelectAll){ angular.forEach($scope.allOptions,function(option){ $scope.yesOptions.push(option.client) }) $scope.textSelectAll = "已选择"+$scope.yesOptions.length+"条"; }else{ $scope.textSelectAll = "全部选择"; } $timeout(function(){ $scope.isClickSelectAll = false; },300) }; $scope.getOptions = function (event) { if(event.target.className === "onlyQC")return; $scope.isShowOptions = !$scope.isShowOptions; }; //组件失焦时,执行下面函数。想一想,什么时候组件聚焦? $scope.getBlurOut = function () { $timeout(function(){ if(!$scope.isFocusInput){ $scope.isShowOptions = false; } }) }; $scope.getFocusInput = function () { $scope.isFocusInput = true; }; //输入框失焦时,执行下面函数 $scope.getBlurInput = function () { $scope.isFocusInput = false; $timeout(function(){ if(!$scope.isClickOptions && !$scope.isClickSelectAll){ $scope.isShowOptions = false; } },200) }; $scope.clickOptions = function (option) { var flag = true; $scope.isClickOptions = true; angular.forEach($scope.yesOptions,function(client){ if(option.client===client){ flag = false; $scope.yesOptions.splice($scope.yesOptions.indexOf(client),1); } }); if(flag){ $scope.yesOptions.push(option.client); } $scope.textSelectAll = "已选择"+$scope.yesOptions.length+"条"; $timeout(function(){ $scope.isClickOptions = false; },300) }; $scope.changeInputValue = function () { if($scope.selectedOption){ $scope.isShowSelectAll = false; $scope.filterOptions = []; angular.forEach($scope.allOptions,function(option){ if(option.client.indexOf($scope.selectedOption) != -1){ $scope.filterOptions.push(option); } }); }else{ $scope.isShowSelectAll = true; $scope.filterOptions = angular.copy($scope.allOptions); } }; $scope.protocolType.getValues = function () { var client = []; var server = []; var clientNo = []; var serverNo = []; var obj; angular.forEach($scope.allOptions,function(option){ if($scope.yesOptions.indexOf(option.client) != -1){ client.push(option.client) server.push(option.server) }else{ clientNo.push(option.client) serverNo.push(option.server) } }); if($scope.isSelectAll){ obj = { is_all:1, client:clientNo, server:serverNo, } }else{ obj = { is_all:0, client:client, server:server, } } return obj }; }, link: function (scope, ele, attrs) { } }; }); app.factory('checkImg', function () { return { fork: '', yes: '', no: '' }; }); </script> 2、searchSelectMid <!DOCTYPE html> <html lang="en" ng-app="myModel"> <head> <meta charset="UTF-8"> <title>searchSelectMid</title> <script type="text/javascript" src="https://cdn.bootcss.com/angular.js/1.6.2/angular.js"></script> <style> .outer-div{ width:210px; height:36px; cursor: pointer; } .select-div{ border: 1px solid #dddddd; padding: 5px 10px; background: #fafafa; margin-right: 5px; border-radius: 4px; height:24px; color:#000; } .option-both{ position: absolute; border: 2px solid #dddddd; width:444px; background:#ffffff; z-index: 10; padding:6px } .option-filter{ padding: 5px 10px; background: #fafafa; margin-right: 5px; margin-bottom: 5px; border-radius: 4px; text-align: left; } .option-filter:hover { background: #cccccc !important; } .option-summary{ max-height: 360px; padding: 4px; display: flex; justify-content: flex-start; flex-wrap: wrap; overflow-y: auto; } .yes-selected{ background:#237eff; border-radius:4px; color:#ffffff } .no-selected{ background:#ffffff; border-radius:0; border-bottom:1px solid #e7e7e7; color:#000000 } </style> </head> <body> <div ng-controller="thisCtrl"> <button ng-click="getAllSelectedOptions()">点击获取被选中项</button> <duv>{{result}}</duv> <search-select-mid protocol-type="protocolType"></search-select-mid> </div> </body> </html> <script> var app = angular.module('myModel', []); app.controller('thisCtrl', function ($scope) { $scope.protocolType = { "yesOptions": [ "c1222", "cnip", "dnp3", "dns", "doip", "egd", "enip", "enip-io", "fins", "ftp", "ftp-data", "ge-srtp", "goose", "hartip", "http", "iec104", "knxip", "mms", "modbus", "opcda", "opcua", "pn-dcp", "pn-mrrt", "pn-ptcp", "pop3", "rdp", "s7", "s7-plus", "sercos", "smb", "smb2", "smtp", "sv", "telnet" ], "allOptions": [ { "server": "c1222", "client": "c1222" }, { "server": "cnip", "client": "cnip" }, { "server": "dnp3", "client": "dnp3" }, { "server": "dns", "client": "dns" }, { "server": "doip", "client": "doip" }, { "server": "egd", "client": "egd" }, { "server": "enip", "client": "enip" }, { "server": "enip-io", "client": "enip-io" }, { "server": "fins", "client": "fins" }, { "server": "ftp", "client": "ftp" }, { "server": "ftp-data", "client": "ftp-data" }, { "server": "ge-srtp", "client": "ge-srtp" }, { "server": "goose", "client": "goose" }, { "server": "hartip", "client": "hartip" }, { "server": "http", "client": "http" }, { "server": "iec104", "client": "iec104" }, { "server": "knxip", "client": "knxip" }, { "server": "mms", "client": "mms" }, { "server": "modbus", "client": "modbus" }, { "server": "opcda", "client": "opcda" }, { "server": "opcua", "client": "opcua" }, { "server": "pn-dcp", "client": "pn-dcp" }, { "server": "pn-mrrt", "client": "pn-mrrt" }, { "server": "pn-ptcp", "client": "pn-ptcp" }, { "server": "pop3", "client": "pop3" }, { "server": "rdp", "client": "rdp" }, { "server": "s7", "client": "s7" }, { "server": "s7-plus", "client": "s7-plus" }, { "server": "sercos", "client": "sercos" }, { "server": "smb", "client": "smb" }, { "server": "smb2", "client": "smb2" }, { "server": "smtp", "client": "smtp" }, { "server": "sv", "client": "sv" }, { "server": "telnet", "client": "telnet" } ], "isFull": true } $scope.getAllSelectedOptions = function(){ var values = $scope.protocolType.getValues(); $scope.result = JSON.stringify(values) }; }); app.directive('searchSelectMid', function () { var html = ` <div tabindex="-1" class="outer-div" ng-blur="outerBlur()" ng-click="outerFocus()" ng-mouseleave="outerLeave()" style=" width:210px; height:36px; cursor: pointer;padding:0"> <!-- 以下是select --> <div ng-mouseenter="enterSelect()" ng-click="clickSelect()" style="padding:4px 0 0 10px;height:34px;" class="select-div"> <span ng-show="yesOptions[0]" ng-bind="yesOptions[0]"></span> <span ng-show="yesOptions.length>1" ng-bind="' ......'"></span> <span ng-show="yesOptions.length==0"> <span style="color:#87a6c9;">请单选或多选或滤选</span> </span> </div> <!-- 以下是option --> <div style="position: relative;" ng-show="isShowOptions||isShowSummary"> <div style="position: absolute;border: 1px solid #818080;width:300px;background:#ffffff;z-index: 10;padding:6px" ng-show="!isShowSummary"> <div> <input type="text" style="width:156px;height:32px;border-radius:4px;margin:4px 0;padding-left: 8px;" placeholder="请输入过滤条件" ng-change="changeInputValue()" ng-model="selectedOption" /> <span style="padding-bottom: 5px;display:inline-block;width:112px;" ng-style="{'visibility': isShowSelectAll? 'visible':'hidden'}"> <img ng-click="selectAll()" src="{{isSelectAll?checkImg.yes:checkImg.no}}" width="16px" height="16px"/> <span>{{textSelectAll}}</span> </span> </div> <div style="max-height: 360px;overflow: auto;" class="youSelect-option"> <div ng-repeat="option in filterOptions track by $index" style=" padding: 2px 8px; background: #fafafa; margin-right: 5px; margin-bottom: 1px; text-align: left; " ng-click="clickOptions(option)" ng-style="yesOptions.indexOf(option.client)>-1?{'background':'#237eff','color':'#ffffff'}:{'background':'#ffffff','color':'#495057'}"> <span ng-bind="option.client" style="height:24px;line-height:24px;display: inline-block;"></span> <i class="fa" style="float:right;position: relative;top:5px" ng-show="yesOptions.indexOf(option.client)>-1"></i> </div> </div> </div> <div style="position: absolute;border: 1px solid #818080;width:300px;background:#ffffff;z-index: 10;padding:6px;box-shadow: 10px 10px 5px #888888;" ng-show="isShowSummary && yesOptions.length>0"> <div style=" max-height: 360px; padding: 4px; display: flex; justify-content: flex-start; flex-wrap: wrap; overflow-y: auto; "> <span style="position:relative;top:4px;">{{'已选'+yesOptions.length+'条 '}}</span> <span ng-repeat="client in yesOptions track by $index" style=" padding: 4px; background: #fafafa; margin-right: 5px; " ng-bind="client" style="color: #000" ></span> </div> </div> </div> </div> `; return { restrict: 'E', template: html, scope: { protocolType: '=protocolType',//所有选项 }, replace: true, controller: function ($scope,$timeout,checkImg) { $scope.checkImg = checkImg; $scope.isShowSummary = false; //是否显示已选项汇总 $scope.isShowSelectAll = true; //是否显示全选 $scope.isShowOptions = false; //是否显示下拉项 $scope.isFocus = false; $scope.isSelectAll = $scope.protocolType.isFull||false; //是否全选 $scope.yesOptions = angular.copy($scope.protocolType.yesOptions); $scope.allOptions = angular.copy($scope.protocolType.allOptions); $scope.filterOptions = angular.copy($scope.allOptions); $scope.textSelectAll = "全部选择"; if($scope.isSelectAll){ $scope.textSelectAll = "已选择"+$scope.yesOptions.length+"条"; } $scope.selectAll = function () { $scope.yesOptions = []; $scope.isSelectAll = !$scope.isSelectAll; if($scope.isSelectAll){ angular.forEach($scope.allOptions,function(option){ $scope.yesOptions.push(option.client) }) $scope.textSelectAll = "已选择"+$scope.yesOptions.length+"条"; }else{ $scope.textSelectAll = "全部选择"; } }; $scope.enterSelect = function () { if($scope.isShowOptions){ $scope.isShowSummary = false; }else{ $scope.isShowSummary = true; } }; $scope.clickSelect = function () { $scope.isShowSummary = false; $scope.isShowOptions = !$scope.isShowOptions; }; $scope.outerLeave = function () { $scope.isShowSummary = false; }; $scope.outerFocus = function () { $scope.isFocus = true; $timeout(function(){ $scope.isFocus = false; },200) }; $scope.outerBlur = function () { $timeout(function(){ if(!$scope.isFocus){//域外失焦 $scope.isShowSummary = false; $scope.isShowOptions = false; } },150) }; $scope.clickOptions = function (option) { var flag = true; angular.forEach($scope.yesOptions,function(client){ if(option.client===client){ flag = false; $scope.yesOptions.splice($scope.yesOptions.indexOf(client),1); } }); if(flag){ $scope.yesOptions.push(option.client); } $scope.textSelectAll = "已选择"+$scope.yesOptions.length+"条"; }; $scope.changeInputValue = function () { if($scope.selectedOption){ $scope.isShowSelectAll = false; $scope.filterOptions = []; angular.forEach($scope.allOptions,function(option){ if(option.client.indexOf($scope.selectedOption) != -1){ $scope.filterOptions.push(option); } }); }else{ $scope.isShowSelectAll = true; $scope.filterOptions = angular.copy($scope.allOptions); } }; $scope.protocolType.getValues = function () { var client = []; var server = []; var clientNo = []; var serverNo = []; var obj; angular.forEach($scope.allOptions,function(option){ if($scope.yesOptions.indexOf(option.client) != -1){ client.push(option.client) server.push(option.server) }else{ clientNo.push(option.client) serverNo.push(option.server) } }); if($scope.isSelectAll){ obj = { is_all:1, client:clientNo, server:serverNo, } }else{ obj = { is_all:0, client:client, server:server, } } return obj }; $scope.protocolType.empty = function (array) { $scope.yesOptions = array||[]; $scope.textSelectAll = "全部选择"; $scope.isSelectAll = false; }; $scope.protocolType.full = function () { $scope.isSelectAll = true; $scope.selectedOption = ''; $scope.changeInputValue(); $scope.yesOptions = angular.copy($scope.protocolType.yesOptions); $scope.textSelectAll = "已选择"+$scope.yesOptions.length+"条"; }; }, link: function (scope, ele, attrs) { } }; }); app.factory('checkImg', function () { return { yes: '', no: '' }; }); </script> 3、searchSelectEmpty <!DOCTYPE html> <html lang="en" ng-app="myModel"> <head> <meta charset="UTF-8"> <title>searchSelectEmpty</title> <script type="text/javascript" src="https://cdn.bootcss.com/angular.js/1.6.2/angular.js"></script> <style> .outer-div{ width:300px; height:36px; cursor: pointer; } .select-div{ border:1px solid #cccccc; border-radius: 6px; } .option-outer{ position: absolute; border: 2px solid #dddddd; width:444px; background:#ffffff; z-index: 10; padding:6px } .option-inner{ padding: 5px 10px; background: #fafafa; margin-right: 5px; margin-bottom: 5px; border-radius: 4px; text-align: left; border-bottom: 1px solid #e7e7e7; } .option-inner:hover { background: #cccccc !important; } .form-input{ width:180px; height:34px; border-radius: 16px; border:1px solid #ffffff; font-size: 16px; padding-left: 20px; } .form-input:focus { outline:none } </style> </head> <body> <div ng-controller="thisCtrl"> <button ng-click="getAllSelectedOptions()">点击获取被选中项</button> <duv>{{result}}</duv> <search-select-empty server-value="serverValue" protocol-type="protocolType"></search-select-empty> </div> </body> </html> <script> var app = angular.module('myModel', []); app.controller('thisCtrl', function ($scope) { $scope.serverValue = ''; $scope.protocolType = [ { "server": "c1222", "client": "c1222" }, { "server": "cnip", "client": "cnip" }, { "server": "dnp3", "client": "dnp3" }, { "server": "dns", "client": "dns" }, { "server": "doip", "client": "doip" }, { "server": "egd", "client": "egd" }, { "server": "enip", "client": "enip" }, { "server": "enip-io", "client": "enip-io" }, { "server": "fins", "client": "fins" }, { "server": "ftp", "client": "ftp" }, { "server": "ftp-data", "client": "ftp-data" }, { "server": "ge-srtp", "client": "ge-srtp" }, { "server": "goose", "client": "goose" }, { "server": "hartip", "client": "hartip" }, { "server": "http", "client": "http" }, { "server": "iec104", "client": "iec104" }, { "server": "knxip", "client": "knxip" }, { "server": "mms", "client": "mms" }, { "server": "modbus", "client": "modbus" }, { "server": "opcda", "client": "opcda" }, { "server": "opcua", "client": "opcua" }, { "server": "pn-dcp", "client": "pn-dcp" }, { "server": "pn-mrrt", "client": "pn-mrrt" }, { "server": "pn-ptcp", "client": "pn-ptcp" }, { "server": "pop3", "client": "pop3" }, { "server": "rdp", "client": "rdp" }, { "server": "s7", "client": "s7" }, { "server": "s7-plus", "client": "s7-plus" }, { "server": "sercos", "client": "sercos" }, { "server": "smb", "client": "smb" }, { "server": "smb2", "client": "smb2" }, { "server": "smtp", "client": "smtp" }, { "server": "sv", "client": "sv" }, { "server": "telnet", "client": "telnet" } ]; $scope.getAllSelectedOptions = function(){ $scope.result = $scope.serverValue }; }); app.directive('searchSelectEmpty', function () { var html = ` <div class="outer-div" > <!-- 以下是select --> <div ng-click="clickSelect()" class="select-div"> <input type="text" class="form-input" ng-blur="getInputBlur()" ng-change="changeInputValue()" ng-model="clientValue" /> <img src="{{checkImg.triangle}}" width="12px" height="6px" style="float:right;position: relative;top:16px;left:-6px" /> </div> <!-- 以下是option --> <div style="position: relative;" ng-show="isShowOptions && filterOptions.length"> <div class="option-outer" > <div style="max-height: 360px;overflow: auto;padding-left: 6px;"> <div ng-repeat="option in filterOptions track by $index" class="option-inner" ng-click="clickOptions(option)"> <span ng-bind="option.client" style="height:34px;line-height:34px;display: inline-block;"></span> </div> </div> </div> </div> </div> `; return { restrict: 'E', template: html, scope: { protocolType: '=protocolType',//所有选项 serverValue: '=serverValue', }, replace: true, controller: function ($scope,$timeout,checkImg) { $scope.checkImg = checkImg; $scope.isShowOptions = false; $scope.filterOptions = angular.copy($scope.protocolType); $scope.copyOptions = angular.copy($scope.protocolType); $scope.clickSelect = function () { $scope.isShowOptions = true; }; $scope.getInputBlur = function () { $timeout(function(){ $scope.isShowOptions = false; },150) }; $scope.clickOptions = function (option) { $scope.serverValue = option.server; $scope.clientValue = option.client; }; $scope.changeInputValue = function () { if($scope.clientValue){ $scope.filterOptions = []; angular.forEach($scope.copyOptions,function(option){ if(option.client.indexOf($scope.clientValue) != -1){ $scope.filterOptions.push(option); } }); }else{ $scope.filterOptions = angular.copy($scope.copyOptions); } }; }, link: function (scope, ele, attrs) { } }; }); app.factory('checkImg', function () { return { triangle: '', }; }); </script> 六、input-select树结构(嵌套结构,不可以拖拽,左侧没有竖线) 1、主要需求描述 (1)页面初始时,前端自己显示一个空下拉框 (2)点击下拉框,前端向后台发送请求,将返回的数据,既渲染成下拉项,又存为备份 (3)点击其中一个下拉项,该下拉项如果没有下一级或下一级只有输入框,那么直接渲染,如果有下一级且下一级有下拉框,那么带上该下拉项数据,向后台发送请求,将返回的数据分别渲染成多个下拉框和输入框,作为下拉框的子级并缩进 (4)下拉框有的可以多选,有的只能单选,有的0选中时可以输入,叉掉选中项时,对应的子级也一起删掉 (5)以后重复(2)(3)(4),只是要携带自身至树根所有层级的label和value... (6)点击“保存”,将整个树图和层级关系发给后台,后台生成一条数据,前端刷新页面,该条数据出现在表格中 (7)点击该条数据后面的“详情”,前端将该条数据展示成“保存”前的树图,可以继续编辑 (8)另外,点击域内下拉框,下拉框隐现交替,点击域外,域内下拉框隐藏 2、主要需求实现思路 (1)弃用select标签,使用div标签模拟select标签的效果,给div标签添加tabindex属性,使其能够获取和失去焦点 (2)数据和组件都用嵌套结构,见下面“数据示例”和“组件html代码”,以便实现各种展示效果, (3)点击“保存”时,用递归遍历整个嵌套树,提取出后台所需数据 (4)点击“详情”时,用递归遍历本条详情,逐级逐条生成数据,每条数据的生成,都包含:A、本条数据加工getThisChildren(children,detail1),B、本条选项获取并加工getThisOptions(thisChild,detail2),C、下级数据获取getNextChildren(thisChild,detail3);getNextChildren最后将thisChild.children和detail3传给getThisChildren,开启下一级数据的生成;这样纵使后发出的请求先返回数据,也不会出现混乱。 附、数据示例 var childrenTemp = {//下拉框示例 type: type,//决定渲染成下拉框还是输入框 parentOption: formData.selectFront,//本条父级的前端文字, selectLabel: key,//标签内容 selectFront: '',//本条的前端文字,传给本条的子级 selectBack: '',//本条的后台数据,传给本条的子级 selectModel: '',//0选中时输入的数据 type_detail: type_detail,//0选中时,是否禁止输入 protocol_type: parentProtocolType,//是否把本条后代的根定在此处 allOptions: [],//本条所有的下拉项,每项包含前后端数据 yesOptions: [],//本条所有的选中项,每项只包含前端数据 selectChainThis: parentProtocolType? [thisParams] : selectChainThis,//自身父级至树根所有层级的{label:value}对象组成的数组 isFinal: isFinal,//本条是否有子级 isArray: isArray,//本条是否可以多选 isShowOptions: false,//是否显示本条的下拉项 children: [],//本条的所有子级 }; var childrenTemp = {//输入框示例 type: type,//决定渲染成下拉框还是输入框 parentOption: formData.selectFront,//本数据与父级select的哪个选中项关联, inputLabel: key,//标签内容 inputValue: '',//输入的数据 type_detail: type_detail,//是否禁止输入 }; 3、实例1,不可预览,供实用参考 (1)组件html代码 <div style="margin-left: 20px"><!-- 子级缩进 --> <div ng-show="requestParams.which==='addOrEdit'"> <div ng-repeat="formData in formDatas track by $index"> <div ng-if="formData.type==='input'" style="margin: 10px 0"> </div> <div ng-if="formData.type==='select'"> <input-select ng-if="formData.children.length>0" form-datas="formData.children" request-params="requestParams" ></input-select> </div> </div> </div> <div ng-show="requestParams.which==='detail'"> <div ng-repeat="formData in formDatas track by $index"> <div ng-if="formData.type==='input'" style="margin: 10px 0"> </div> <div ng-if="formData.type==='select'"> <input-select ng-if="formData.children.length>0" ></input-select> </div> </div> </div> </div> (2)页面html代码 <input-select ng-if="formDatas.length>0" form-datas="formDatas" request-params="requestParamsAddOrEdit" > </input-select> (3)组件js代码 (function () { angular.module('common-dir').directive('inputSelect', function () { return { restrict: 'E', templateUrl: 'vendor/common-directive/input-select.html', scope: { formDatas: '=formDatas', requestParams: '=requestParams' }, replace: true, controller: function ($scope, auditApi, $timeout) { $scope.getBlur = function (formData, event) { $timeout(function () { formData.isShowOptions = false; }, 200); }; //这个函数用于:向后台发送请求,获取select的option $scope.getOptions = function (formData, event) { //点击select框,获取下拉项 if (event.target.tagName === 'INPUT') return; //当点击select框内部的input框时,阻止 if (event.target.className === 'onlyQC') return; //当点击select框内部的备选项时,阻止 if (formData.allOptions && formData.allOptions.length > 0) { if (event.target.parentElement.className === 'isShowOptions') { formData.isShowOptions = !formData.isShowOptions; } return; //当下拉项已存在时,阻止 } ................................ }; //这个函数用于:叉掉已选中项时,除了删掉自身,还删掉下面对应的所有子项 $scope.optionDelete = function (formData, front) { formData.yesOptions.splice(formData.yesOptions.indexOf(front), 1); for (var i = 0; i < formData.children.length; i++) { if (formData.children[i].parentOption == front) { formData.children.splice( formData.children.indexOf(formData.children[i]), 1 ); i--; } } }; //这个函数用于:点击下拉项中的某项时 $scope.optionClick = function (formData, option) { var isTrue = formData.type_detail === 'protogo' || formData.type_detail === 'object' || formData.isArray; if (!isTrue) { formData.children = []; } ................................ }; }, link: function (scope, ele, attrs) {} }; }).factory('inputSelect', function (auditApi) { return { /*方法说明 *@method getThisChildren 生成本条的全部子级 *@param{Array}selectChainThis 参数链,向后台发送请求前,应当拼接成字符串 *@param{String}label label标签文字 *@param{String}parent 本条来自于父级的被选中项parent *@param{Object}valueOut 所有子级的原始数据 *@param{Array}children 所有子级的最终数据 *@param{Object,Array,String}thisDetail 本条数据的详情,随着递归的深入,越来越短 *@param{Boolean}isLeap 是否需要跳转,有时随着递归的深入,下一级回到了第一级(协议名称),这种情况叫跳转 *@return {undefined} undefined,没有返回值 */ getThisChildren:function ( selectChainThis, label, parent, valueOut, children, thisDetail, isLeap, request ) { if (type === 'input') { thisChild = {}; } else if (type === 'select') { thisChild = {}; if (isFinal === 1) { //没有子级 } else if (isFinal === 0) { //有子级 if (thisDetail.constructor === Object) { var detailKeysArray = Object.keys(thisDetail); for (var i = 0; i < detailKeysArray.length; i++) { var detailKey = detailKeysArray[i]; if (detailKey == label) { var nextDetail = thisDetail[detailKey]; that.getThisOptions(...); } } } else { //点击下拉框时,组件向后台发送请求,然后才出现下拉项,不用在这里获取下拉项。 } } } children.push(thisChild); }, /*方法说明 *@method getThisOptions 生成本子级的所有下拉项 *@param{Object}thisChild 一个子级 *@param{Object,Array,String}nextDetail 本条数据的详情,随着递归的深入,越来越短 *@param{Array}selectChainThis 参数链,向后台发送请求前,应当拼接成字符串 *@param{String}label label标签文字 *@param{Boolean}isLeap 是否需要跳转,有时随着递归的深入,下一级回到了第一级(协议名称),这种情况叫跳转 *@return {undefined} undefined,没有返回值 */ getThisOptions:function ( thisChild, nextDetail, selectChainThis, label, isLeap, request ) { $http({ method: request.method, root: request.root, url: request.url, params: params }) .then(function (result) { allOptions.push(oneOption); for (var j = 0; j < allOptions.length; j++) { var optionOne = allOptions[j]; that.getNextChildren(...); } }); }, /*方法说明 *@method getNextChildren 获取子级的子级 *@param{Object}optionOne 子级的一个下拉项 *@param{Object,Array,String}nextDetail 本条数据的详情,随着递归的深入,越来越短 *@param{Array}selectChainThis 参数链,向后台发送请求前,应当拼接成字符串 *@param{Object}thisChild 一个子级 *@param{Object}params 请求参数 *@param{Boolean}isLeap 是否需要跳转,有时随着递归的深入,下一级回到了第一级(协议名称),这种情况叫跳转 *@return {undefined} undefined,没有返回值 */ getNextChildren:function ( optionOne, nextDetail, selectChainThis, thisChild, params, isLeap, request ) { var that = this; var isFinal = optionOne.isFinal; var parent = optionOne.front; if (that.wholeKeysArray.indexOf(parent) > -1) {//以下是跳转逻辑 $http({ method: request.method, root: request.root, url: request.url, params: { protocol_type: parent } }) .then(function (result) { var selectChainThis = [{ 协议名称: parent }]; //遇到跳转逻辑时,给选择链重新赋值 for (var label in valueOut) { that.getThisChildren(...); } }); } } else {//以下是正常逻辑 if (isFinal === 1) { var resultKey = Object.keys(thisChild.copyOptions)[0]; var valueOut = thisChild.copyOptions[resultKey]; var isObject = nextDetail.constructor === Object; var detailKeysArray; if (isObject) { detailKeysArray = Object.keys(nextDetail); if (detailKeysArray.indexOf(parent) > -1) { for (var label in valueOut) { if (label === parent) { that.getThisChildren(...); thisChild.yesOptions.push(parent); } } } } } else if (isFinal === 0) { var isObject = nextDetail.constructor === Object; if (isObject) { var detailKeysArray = Object.keys(nextDetail); if (detailKeysArray.indexOf(parent) > -1) { $http({ method: request.method, root: request.root, url: request.url, params: thisParams }) .then(function (result) { if (detailKeysArray.indexOf(resultKeysArray[0]) > -1) {//父级的parentOption和子级的label一样 for (var label in valueOut) { that.getThisChildren(...); } } else {//如果出现parentOption和它子级的label不一样,那么根据数据结构,逻辑写在此处。 for (var label in valueOut) { that.getThisChildren(...); } } }); } } } } }, }; }); })(); (4)页面js代码 $scope.formDatas = [ { type: 'select', selectLabel: '协议名称', selectFront: '', selectBack: '', allOptions: [], yesOptions: [], selectChainThis: [], children: [], isFinal: 0 } ]; 4、实例2,可预览,供演示 <!DOCTYPE html> <html lang="en" ng-app="myModel"> <head> <meta charset="UTF-8"> <title>input和select</title> <script type="text/javascript" src="https://cdn.bootcss.com/angular.js/1.6.2/angular.js"></script> </head> <body> <div ng-controller="thisCtrl"> <div> <p>=====可预览=====</p> <p>1、向后台发送请求,获取返回数据,填充$scope.formDatas</p> <p>2、向后台发送请求,获取返回数据,填充$scope.formDatas</p> </div> <input-select ng-if="formDatas.length>0" form-datas="formDatas"> </input-select> </div> </body> </html> <script> var app = angular.module('myModel', []); app.controller('thisCtrl', function ($scope/*,cyRequest*/) { //此处,向后台发送请求,获取返回数据,填充$scope.formDatas $scope.formDatas = [ { type: 'input', inputLabel: '输入框', inputValue: '', },{ type: 'select', selectLabel: '选择框标签', selectValue: '后台1', selectOptions: [ { back: '后台0', front: '前端0' }, { back: '后台1', front: '前端1' }, { back: '后台2', front: '前端2' }, { back: '后台3', front: '前端3' }, { back: '后台4', front: '前端4' }, { back: '后台5', front: '前端5' }, ], selectChainThis: [],//默认值。 selectChainNext: null, children: [],//子级。 },{ type: 'select', selectLabel: '选择框标签2222', selectValue: '后台1', selectOptions: [ { back: '后台0', front: '前端0' }, { back: '后台1', front: '前端1' }, { back: '后台2', front: '前端2' }, { back: '后台3', front: '前端3' }, { back: '后台4', front: '前端4' }, { back: '后台5', front: '前端5' }, ], selectChainThis: [],//默认值。 selectChainNext: null, children: [],//子级。 } ]; }); app.directive('inputSelect', function () { var html = ` <div style="margin-left:10px;"> <div ng-repeat="formData in formDatas track by $index"> <div ng-if="formData.type==='input'" style="margin:10px 0"> <label ng-bind="formData.inputLabel+':'" style="width:200px;text-align:right;display:inline-block"></label> <input ng-model="formData.inputValue" type="text" style="width:300px;border:1px solid gray;height:26px;border-radius:4px;"/> </div> <div ng-if="formData.type==='select'" style="margin:10px 0"> <label ng-bind="formData.selectLabel+':'" style="width:200px;text-align:right;display:inline-block"></label> <select ng-model="formData.selectValue" ng-options="option.back as option.front for option in formData.selectOptions" ng-change="clickOption(formData)" style="width:306px;border:1px solid gray;height:26px;border-radius:4px;" ></select> <input-select ng-if="formData.children.length>0" form-datas="formData.children"></input-select> </div> </div> </div> `; return { restrict: 'E', template: html, scope: { formDatas: '=formDatas' }, replace: true, controller: function ($scope) {//此处注入“公共请求服务” $scope.clickOption = function (formData) { var objOuter={}; objOuter[formData.selectLabel] = formData.selectValue; formData.selectChainNext = angular.copy(formData.selectChainThis); formData.selectChainNext.push(objOuter); //此处,把formData.selectChainNext作为“公共请求服务”的参数,向后台发送请求,获取返回数据result并加工,然后填充formData.children var result=[ { type:'input', label:'输入框' },{ type:'select', label:'选择框标签' },{ type:'select', label:'选择框标签2' } ]; angular.forEach(result,function(value,index){ var objInner = { }; if(value.type==='input'){ objInner.type = value.type; objInner.inputLabel = value.label; objInner.inputValue = ''; }else if(value.type==='select'){ objInner.type = value.type; objInner.selectLabel = value.label; objInner.selectValue = '后台1'; objInner.selectOptions = [ { back: '后台0', front: '前端0' }, { back: '后台1', front: '前端1' }, { back: '后台2', front: '前端2' }, { back: '后台3', front: '前端3' }, { back: '后台4', front: '前端4' }, { back: '后台5', front: '前端5' }, ]; objInner.selectChainThis = angular.copy(formData.selectChainNext); objInner.selectChainNext = null; objInner.children = []; if(index===1){ console.log(objInner.selectChainThis) } } formData.children.push(objInner) }) }; }, link: function (scope, ele, attrs) { } }; }) </script> 七、dir-tree树结构(平行结构,可以拖拽,左侧有竖线) 1、主要需求描述 (1)页面初始时,初始化组件,向后台发送请求,获取嵌套数据,改成平级数据,计算并添加缩进和左侧线,渲染组件 (2)可以编辑本条数据 (3)可以拖拽本条数据至目标数据上,将自身和目标数据发给后台,最后刷新页面,获取新嵌套数据(被拖拽数据及其后代成为目标数据的后代) 2、主要需求实现思路 (1)拖拽效果用插件 (2)递归实现嵌套数据改成平级数据,非如此则拖拽出错 (3)递归实现左侧竖线 附、数据示例 var obj={ id: item.id,//本级id name: item.name, puuid: item.puuid, uuid: item.uuid, idsArray: item.idsArray,//供判断层级关系时用 isShowSelf: item.collapsed||true,//是否显示自身 //以上为后台返回,以下为前端添加 isShowChild: true,//是否显示后代 leftTime: leftTime,//竖线从左缩进次数 downTime: 0,//竖线向下延伸次数 } (5)给div标签添加tabindex属性,使其能够获取和失去焦点 3、实例1,不可预览,供实用参考 (1)组件html代码 <div> <div style="display:flex; flex-direction: row" ng-repeat="formData in formDatas track by $index" > <div class="dir-tree-line" ng-drag="true" ng-drop="true" ng-show="formData.isShowSelf" ng-style="{'margin-left':formData.leftTime*28+'px'}" ng-drop-success="dropSuccess($index, $data, $event)" ng-drag-data="formData" ng-click="clickItem(formData)" style="position:relative" > <div> <span ng-style="{'background': $index===0? '#ffffff':'rgb(162, 142, 142)'}" style="height:1px;width:16px;display: inline-block;position: relative;top:-4px"></span> <span class="dir-tree-label" ng-class="formData.isHasChild?((selectedItem.showId && selectedItem.showId === formData.uuid)?'dir-tree-haschild-blue':'dir-tree-haschild-black'):((selectedItem.showId && selectedItem.showId === formData.uuid)?'dir-tree-nochild-blue':'dir-tree-nochild-black')" ></span> <span ng-bind="formData.name" ng-class="(selectedItem.showId && selectedItem.showId === formData.uuid)?'dir-tree-label-text-blue':'dir-tree-label-text-black'" ng-drag="true" ng-drag-data="formData" ></span> <span class="dir-tree-label" ng-show="formData.isHasChild && !formData.isShowChild" ng-class="(selectedItem.showId && selectedItem.showId === formData.uuid)?'dir-tree-haschild-tianjia-blue':'dir-tree-haschild-tianjia-black'" ></span> </div> <div style="position:relative;width:1px;background: rgb(162, 142, 142);left:27px;top:-7px;" ng-style="{'height':((formData.downTime+0.5)*36+8)+'px'}" ng-show="formData.isShowChild && formData.isHasChild" ></div> </div> <div style="flex:1;"></div> </div> </div> (2)页面html代码 <dir-tree request-datas='requestDatas' selected-item='selectedAdd' ng-if="decideOther.isIf"></dir-tree> (3)组件js代码 (function () { angular.module('common-dir').directive('dirTree', function () { return { restrict: 'E', templateUrl: 'vendor/common-directive/dir-tree.html', scope: { requestDatas: '=requestDatas',//与前后台交互相关的内容 selectedItem: '=selectedItem',//树图的被选中项 decideOther: '=decideOther'//本树是否决定“其它树”的隐现 }, replace: true, controller: function ($scope, $timeout, auditApi) { if(!$scope.selectedItem.isAbandonDrag){ $scope.selectedItem.callback = function(name,uuid,puuid){ auditApi .query({ method: $scope.requestDatas.method||'put', root: $scope.requestDatas.root||'organization', url: $scope.requestDatas.url||'/info', data: { name:name,//被修改节点的名称 uuid:uuid,//被修改节点的uuid puuid:puuid,//被修改的节点,应移到哪个节点之下 }, pageLoad: true, }) .then(function () { layer.msg('拖拽成功!'); $scope.getDatas(); }).catch(function(){ $scope.getDatas(); }); } }else{ delete $scope.selectedItem.callback } $scope.clickItem = function(formData){ $scope.selectedItem.name = formData.name; $scope.selectedItem.puuid = formData.puuid; $scope.selectedItem.uuid = formData.uuid; $scope.selectedItem.showId = formData.uuid; var id = formData.id; formData.isShowChild = !formData.isShowChild $scope.formDatas.forEach(function(item){ if(item.idsArray.indexOf(id) > -1 && item.idsArray.length > formData.idsArray.length){ item.isShowChild = formData.isShowChild; item.isShowSelf = formData.isShowChild; } }) $scope.addDownTime($scope.formDatas); } $scope.dropSuccess = function(index,data,event){ var toItem = $scope.formDatas[index]; if(angular.isFunction($scope.selectedItem.callback)){ $scope.selectedItem.callback(data.name,data.uuid,toItem.uuid); } } $scope.getDatas = $scope.requestDatas.getDatas = function(isNewAdd){ auditApi .query({ method: $scope.requestDatas.method||'get', root: $scope.requestDatas.root||'organization', url: $scope.requestDatas.url||'/info', pageLoad: true, }) .then(function (res) { var list = angular.copy(res.rows); var lastList = []; //以下给每条数据添加idsArray属性,供判断层级关系时用 function addEachRowIdsArray(list,fatherId){ list.forEach(function(item) { item.idsArray = fatherId.concat(item.id); if(item.children){ addEachRowIdsArray(item.children,item.idsArray) } }); } addEachRowIdsArray(list,[]) //以下把父子嵌套数据改造成兄弟并列数据,不如此则拖拽出错 function getList(list,leftTime){ angular.forEach(list,function(item){ var obj={ id: item.id, name: item.name, puuid: item.puuid, uuid: item.uuid, idsArray: item.idsArray, isShowSelf: item.collapsed||true,//是否显示自身 //以上为后台返回,以下为前端添加 isShowChild: true,//是否显示后代 leftTime: leftTime,//竖线从左缩进次数 downTime: 0,//竖线向下延伸次数 } lastList.push(obj) if(item.children.length){ obj.isHasChild = true; getList(item.children,leftTime+1) } }) } getList(list,1); if(isNewAdd){ $scope.selectedItem.showId = lastList[0].uuid; $scope.selectedItem.uuid = lastList[0].uuid; } $scope.selectedItem.isInit = Boolean(lastList.length); if($scope.decideOther){//不如此,修改后“其它树”不刷新 $scope.decideOther.isIf = false; $timeout(function(){ $scope.decideOther.isIf = true; $scope.formDatas = lastList; $scope.addDownTime($scope.formDatas) }) } $scope.formDatas = lastList;//已经完成加工的树图总数据! $scope.addDownTime($scope.formDatas)//给总数据的每条数据添加向下延伸次数 }); } $scope.addDownTime = function(formDatas){ //以下给formDatas的每一项添加downTime formDatas.forEach(function(item1){ item1.downTime = 0 ; if(!item1.isShowChild)return; //以下添加item1的downTime var id1 = item1.id; var idsArray = []; //以下获取本级的所有子代的id,并存储在idsArray中 formDatas.forEach(function(item2){ if(item2.idsArray.indexOf(id1) > -1 && item2.idsArray.length == item1.idsArray.length+1){ idsArray.push(item2.id) } }) //以下删除idsArray的最后一项 idsArray.pop(); //以下计算本项的downTime for(var i = 0; i<idsArray.length; i++){ var id2 = idsArray[i];//第i个子代的id for(var ii = 0; ii<formDatas.length; ii++){ var item3 = formDatas[ii]; if(item3.idsArray.indexOf(id2)>-1){//从所有数据中,找到第i个子代 item1.downTime++; if(item3.isHasChild && !item3.isShowChild){//第i个子代有子代且没有展开 var id3 = item3.id; formDatas.forEach(function(item4){ if(item4.idsArray.indexOf(id3) > -1 && item4.idsArray.length > item3.idsArray.length){ ii++//跳过没展开的子代 } }) } } } } }) } $scope.getDatas() }, link: function (scope, ele, attrs) {} }; }); })(); (4)页面js代码 //以下是树图属性传参的数据 $scope.decideOther = {//本树是否决定“其它树”的隐现 isIf: true }; $scope.requestDatas = { };//与前后台交互相关的内容 $scope.selectedDatas = {//树图的被选中项 name: '', uuid: '', puuid: '', showId: '', }; 4、实例2,可预览,供演示用 <!DOCTYPE html> <html lang="en" ng-app="myModel"> <head> <meta charset="UTF-8"> <title>passWord</title> <style> .dir-tree-label{ padding-left: 20px; margin-right: 4px; height: 36px; line-height: 36px; background-position: left 4px; background-repeat: no-repeat; cursor:pointer; user-select:none; } .dir-tree-label-text{ cursor: pointer; user-select: none; } .dir-tree-label-black{ color:black } .dir-tree-label-red{ color:red } </style> <script type="text/javascript" src="https://cdn.bootcss.com/angular.js/1.6.2/angular.js"></script> <script src="https://cdn.bootcss.com/ngDraggable/0.1.11/ngDraggable.js"></script> </head> <body> <div ng-controller="thisCtrl"> <dir-tree form-datas="lastList" other-datas='otherDatas'></dir-tree> </div> </body> </html> <script> var app = angular.module('myModel', ['ngDraggable']); app.controller('thisCtrl', function ($scope) { var res = { "rows": [{ "children": [{ "children": [{ "children": [{ "children": [], "isShowSelf": true, "id": "12", "name": "裁员部一", "puuid": "1ec9f238-9aba-484c-b857-d4183200053c", "uuid": "1e8ae51d-8d2f-486e-b1d1-ff1009f8461e" }, { "children": [], "isShowSelf": true, "id": "13", "name": "裁员部二", "puuid": "1ec9f238-9aba-484c-b857-d4183200053c", "uuid": "be521aa9-1e8c-4262-a156-2d02e3cb9c51" } ], "isShowSelf": true, "id": "4", "name": "人力部一", "puuid": "bcb3e97f-d8ef-4e33-a234-09a3fef8533d", "uuid": "1ec9f238-9aba-484c-b857-d4183200053c" }, { "children": [], "isShowSelf": true, "id": "5", "name": "人力部二", "puuid": "bcb3e97f-d8ef-4e33-a234-09a3fef8533d", "uuid": "ba10a598-9af7-47ba-9c34-ee41d85f0525" }, { "children": [], "isShowSelf": true, "id": "6", "name": "人力部三", "puuid": "bcb3e97f-d8ef-4e33-a234-09a3fef8533d", "uuid": "a880024f-e906-4558-833c-4ec7a20927e3" }, { "children": [], "isShowSelf": true, "id": "7", "name": "人力部四", "puuid": "bcb3e97f-d8ef-4e33-a234-09a3fef8533d", "uuid": "25de3bf5-4848-48e5-a161-04d1492a4edb" } ], "isShowSelf": true, "id": "2", "name": "人力部", "puuid": "f2a8c4ea-e270-44d0-a7bd-18c62fb55358", "uuid": "bcb3e97f-d8ef-4e33-a234-09a3fef8533d" }, { "children": [{ "children": [], "isShowSelf": true, "id": "8", "name": "财务部一", "puuid": "44667cbb-a1d5-4187-84aa-c3ee347d873e", "uuid": "c036eab1-0f4d-4732-be3c-8e2da46cb60e" }, { "children": [], "isShowSelf": true, "id": "9", "name": "财务部二", "puuid": "44667cbb-a1d5-4187-84aa-c3ee347d873e", "uuid": "ee22c501-e797-4190-84d7-41c0523ac784" }, { "children": [], "isShowSelf": true, "id": "10", "name": "财务部三", "puuid": "44667cbb-a1d5-4187-84aa-c3ee347d873e", "uuid": "60dd79cf-20b7-4d04-92c0-968eb85620c1" }, { "children": [], "isShowSelf": true, "id": "11", "name": "财务部四", "puuid": "44667cbb-a1d5-4187-84aa-c3ee347d873e", "uuid": "f7a76408-c3ba-4c4b-8f77-d520407beb31" } ], "isShowSelf": true, "id": "3", "name": "财务部", "puuid": "f2a8c4ea-e270-44d0-a7bd-18c62fb55358", "uuid": "44667cbb-a1d5-4187-84aa-c3ee347d873e" } ], "isShowSelf": true, "id": "1", "name": "中国石油", "puuid": "", "uuid": "f2a8c4ea-e270-44d0-a7bd-18c62fb55358" }] }; var list = angular.copy(res.rows); var lastList = []; //以下解决隐藏和显示问题 function addEachRowIds(list,fatherId){ list.forEach(function(item) { item.idsArray = fatherId.concat(item.id); if(item.children){ addEachRowIds(item.children,item.idsArray) } }); } addEachRowIds(list,[]) //以下把父子嵌套数据改造成兄弟并列数据,不如此则拖拽出错,极端重要!!! function getList(list,left){ angular.forEach(list,function(item){ var obj={ id: item.id, name: item.name, puuid: item.puuid, uuid: item.uuid, idsArray: item.idsArray, isShowSelf: item.isShowSelf, left: left, } lastList.push(obj) if(item.children.length){ obj.isHasChild = true; getList(item.children,left+1) } }) } getList(list,1); $scope.lastList = lastList; $scope.otherDatas = {//页面树图的被选中项 name: '',//被选中项的name uuid: '',//被选中项的uuid puuid: '',//被选中项的puuid,即该数据父级的uuid showId: '',//被选中项的showId,取值为“该条数据”的uuid,在<dir-tree>组件中,showId与每条数据的uuid进行比较,相等的展示为红色,为被选中项本身 callback: function(name,uuid,puuid){//拖动结束后,触发的函数 } }; }); app.directive('dirTree', function () { var html = ` <div> <div class="dir-tree-label" ng-drag="true" ng-drop="true" ng-show="formData.isShowSelf" ng-style="{'margin-left':formData.left*20+'px'}" ng-repeat="formData in formDatas track by $index" ng-drop-success="dropSuccess($index, $data, $event)" ng-drag-data="formData" > <img ng-src="{{formData.isHasChild? fileImg.havechild : fileImg.nochild}}" ng-click="clickImg(formData)"> <span class="dir-tree-label-text" ng-bind="formData.name" ng-class="(otherDatas.showId && otherDatas.showId === formData.uuid)?'dir-tree-label-red':'dir-tree-label-black'" ng-click="clickText(formData)" ng-drag="true" ng-drag-data="formData" ></span> </div> </div> `; return { restrict: 'E', template: html, scope: { formDatas: '=formDatas', otherDatas: '=otherDatas'//页面树图的被选中项 }, replace: true, controller: function ($scope,fileImg) { $scope.fileImg = fileImg; $scope.clickImg = function(formData){ $scope.clickText(formData); var thisArrayIds = formData.idsArray; $scope.formDatas.forEach(function(item){ var flag = true; var itemArrayIds = item.idsArray; thisArrayIds.forEach(function(id,index){ if(itemArrayIds.indexOf(id) === -1){ flag = false; } }) if(flag){ item.isShowSelf = !item.isShowSelf } }) formData.isShowSelf = true } $scope.clickText = function(formData){ $scope.otherDatas.name = formData.name; $scope.otherDatas.puuid = formData.puuid; $scope.otherDatas.uuid = formData.uuid; $scope.otherDatas.showId = formData.uuid; } $scope.dropSuccess = function(index,data,event){ /** * 函数说明:拖拽结束时触发的函数 * @index:目标所在位置的序号 * @data:初始位置的数据 * @event:初始位置的鼠标事件 * */ var toItem = $scope.formDatas[index]; var fromIndex = $scope.formDatas.findIndex(function(item){ return Object.is(data.uuid ,item.uuid) }) $scope.formDatas[index] = data; $scope.formDatas[fromIndex] = toItem; $scope.otherDatas.callback(data.name,data.uuid,toItem.uuid); } }, link: function (scope, ele, attrs) {} }; }); app.factory('fileImg', function () { return { nochild: '', havechild: '', }; }); </script>