19、angular1之pass-word组件、input-select组件 、only-select组件(两种)、on-off组件(缺八)

122 阅读2分钟

19、angular1之pass-word组件、input-select组件 、only-select组件(两种)、on-off组件、layui中的datetime示例、京东购物车、两种作用域绑定、两种函数传递

一、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+)*$/,//邮箱验证
      chineseCharacters: /^[\u4e00-\u9fa5]{0,}$/,//汉字验证
      blankSpace: /^\s*$/ //零到多个空格
    };
  });
  app.factory('servicePicture', function () {
    return {
      eyeClose: '',
      eyeOpen: '',
      correct: '',
      error: '',
      triangle: ''
    };
  });
</script>

二、input-select(自定义输入框和下拉框,可预览)
使用场景:有个select标签,选择其中的一项,向后台发送请求,然后根据后台返回的数据,渲染input或select标签,如果是select,还可以继续选择。。。
<!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>

三、only-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;">
      <only-select-up form-datas-up="formDatasUp" current-datas-up="currentDatasUp" two-label-up="twoLabelUp"></only-select-up>
    </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;">
      <only-select-down form-datas-down="formDatasDown" current-datas-down="currentDatasDown" two-label-down="twoLabelDown"></only-select-down>
    </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('onlySelectUp', 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('onlySelectDown', 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
        }
        $scope.thisSwitch.callback()
      };
    },
    link: function (scope, ele, attr) {
      
    }
  };
});
app.factory('onOff', function () {
  return {
    yes: '',
    no: '',
  };
})
</script>
 
五、angular1里用layui中的datetime示例
来源:https://www.layui.com/doc/modules/laydate.html#type
<dir-datepicker date-model="pagin_init.filter_option.beginTime"></dir-datepicker>
(function () {
  angular
    .module('common-dir')
    .directive('dirDatepicker', function () {
      return {
        restrict: 'E',
        template: '<input  type="text" class="form-control common-select-time" ng-model="dateModel"  id="{{dateInputId}}"  style="width:214px" ng-disabled="isDisable" />',
        replace: true,
        scope: {
          format: '@',
          dateModel: '=',
          isDisable: '=',
          callback: '&'
        },
        controller: function ($scope) {
          $scope.dateInputId = 'id' + Math.random();
          angular.element(document).ready(function () {
            var config = {
              elem: '#' + $scope.dateInputId,
              theme: '#007bff',
              trigger: 'click',
              done: function (value) {
                $scope.dateModel = value;
                angular.isFunction($scope.callback) ? $scope.callback() : '';
              }
            };
            if ($scope.format) {
              config.format = $scope.format;//自定义格式
            } else {
              config.type = 'datetime';//控件选择类型
            }
            laydate.render(config);
          });
        },
        link: function () {
        }
      };
    });
})();
控件选择完毕后的回调
点击日期、清空、现在、确定均会触发。回调返回三个参数,分别代表:生成的值、日期时间对象、结束的日期时间对象
laydate.render({
  elem: '#test',
  done: function(value, date, endDate){
    console.log(value); //得到日期生成的值,如:2017-08-18
    console.log(date); //得到日期时间对象:{year: 2017, month: 8, date: 18, hours: 0, minutes: 0, seconds: 0}
    console.log(endDate); //得结束的日期时间对象,开启范围选择(range: true)才会返回。对象成员同上。
  }
});

六、agular1实现京东购物车
<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <title>Title</title>
  <link rel="stylesheet" href="https://cdn.bootcss.com/bootstrap/4.1.1/css/bootstrap.css">
  <script type="text/javascript" src="https://cdn.bootcss.com/angular.js/1.5.8/angular.js"></script>
</head>
<body ng-app="myApp" ng-controller="myCtrl" style="margin:20px;" ng-cloak >
  <div>
    <label>全选</label>
    <input type="checkbox" ng-model="allCheck" ng-click="allChecked()">
    <span>总金额:{{ totalMoney() | currency:"¥"}}</span>
  </div>
  <table class="table table-bordered" ng-repeat="oneShop in allShops track by $index"  ng-init="outerIndex = $index" style="margin:30px auto">
    <thead>
    <tr>
      <th><input type="checkbox" ng-model="oneShop.checked" ng-click="shopChecked(oneShop,allShops)"></th>
      <th colspan="7">{{oneShop.shopName}}</th>
    </tr>
    </thead>
    <tbody>
      <tr ng-repeat="oneGoods in oneShop.shopGoods track by $index"  ng-init="innerIndex = $index">
        <td><input type="checkbox" ng-model="oneGoods.checked" ng-click="singleGoods(oneGoods,oneShop.shopGoods,oneShop,allShops)"></td>
        <td>{{oneGoods.goodsName}}</td>
        <td>{{oneGoods.price}}元/件,共</td>
        <td><span ng-click="add(oneGoods,oneShop.shopGoods,oneShop,allShops)" style="display: inline-block;width:25px;text-align: center; border:1px solid gray;user-select: none;cursor: pointer;">+</span></td>
        <td>{{oneGoods.number=oneGoods.number||1}}</td>
        <td><span ng-click="reduce(oneGoods,oneShop.shopGoods,oneShop,allShops)" style="display: inline-block;width:25px;text-align: center; border:1px solid gray;user-select: none;cursor: pointer;">-</span></td>
        <td>件,本商品共: {{oneGoods.singleMoney}}元</td>
        <td ng-click="delete(innerIndex,outerIndex,oneShop,allShops)" style="user-select: none;cursor: pointer;">删除</td>
      </tr>
    </tbody>
</table>
</body>
</html>
<script>
var myApp = angular.module("myApp", []);
myApp.controller("myCtrl", function ($scope) {
  $scope.allShops = [
    {
      shopName: "专卖店一:北京鸡",
      shopGoods: [
        {
          goodsName: "北京鸡1",
          picture: "images/allShops_01.jpg",
          price: 150.0,
          singleMoney: 150.0,
        },
        {
          goodsName: "北京鸡2",
          picture: "images/allShops_02.jpg",
          price: 119.0,
          singleMoney: 119.0,
        },
        {
          goodsName: "北京鸡3",
          picture: "images/allShops_03.jpg",
          price: 101.0,
          singleMoney: 101.0,
        },
      ],
    },
    {
      shopName: "专卖店二:北京鸭",
      shopGoods: [
        {
          goodsName: "北京鸭1",
          picture: "images/allShops_04.jpg",
          price: 89.0,
          singleMoney: 89.0,
        },
        {
          goodsName: "北京鸭2",
          picture: "images/allShops_05.jpg",
          price: 99.0,
          singleMoney: 99.0,
        },
      ],
    },
    {
      shopName: "专卖店三:北京鹅",
      shopGoods: [
        {
          goodsName: "北京鹅1",
          picture: "images/allShops_06.jpg",
          price: 289.0,
          singleMoney: 289.0,
        },
      ],
    },
  ];
  //点击加减按钮,数量加减;点击删除按钮,删除商品
  $scope.reduce = function (goods, allGoodsOfOneShop, oneShop, allShops) {
    goods.number--;
    if (goods.number <= 1) goods.number = 1;
    goods.singleMoney = goods.price * goods.number;
    goods.checked = true;
    $scope.singleGoods(goods, allGoodsOfOneShop, oneShop, allShops);
  };
  $scope.add = function (goods, allGoodsOfOneShop, oneShop, allShops) {
    goods.number++;
    goods.singleMoney = goods.price * goods.number;
    goods.checked = true;
    $scope.singleGoods(goods, allGoodsOfOneShop, oneShop, allShops);
  };
  //删除该件商品
  $scope.delete = function (innerIndex, outerIndex, oneShop, allShops) {
    console.log(innerIndex);
    console.log(outerIndex);
    console.log(oneShop);
    console.log(allShops);
    oneShop.shopGoods.splice(innerIndex, 1);
    if (oneShop.shopGoods.length <= 0) {
      allShops.splice(outerIndex, 1);
    }
  };
  /*所有商品总金额计算*/
  $scope.totalMoney = function () {
    var total = 0;
    angular.forEach($scope.allShops, function (outerItem) {
      angular.forEach(outerItem.shopGoods, function (innerItem) {
        if (innerItem.checked) {
          total += innerItem.price * innerItem.number;
        }
      });
    });
    return total;
  };
  /*单件商品选择*/
  $scope.singleGoods = function (
    oneGoods,
    allGoodsOfOneShop,
    oneShop,
    allShops
  ) {
    var flag = true;
    if (oneGoods.checked) {
      angular.forEach(allGoodsOfOneShop, function (innerItem) {
        if (
          innerItem.checked == false ||
          typeof innerItem.checked == "undefined"
        ) {
          flag = false;
        }
      });
    } else {
      $scope.allCheck = false;
      oneShop.checked = false;
      flag = false;
    }
    if (flag) {
      oneShop.checked = true;
      angular.forEach(allShops, function (outerItem) {
        if (
          outerItem.checked == false ||
          typeof outerItem.checked == "undefined"
        ) {
          flag = false;
        }
      });
    }
    if (flag) {
      $scope.allCheck = true;
    }
  };
  /*单家商铺选择*/
  $scope.shopChecked = function (oneShop, allShops) {
    if (oneShop.checked) {
      var flag = true;
      angular.forEach(oneShop.shopGoods, function (innerItem) {
        innerItem.checked = true;
      });
      angular.forEach(allShops, function (outerItem) {
        console.log(outerItem.checked);
        if (
          outerItem.checked == false ||
          typeof outerItem.checked == "undefined"
        ) {
          flag = false;
        }
      });
      if (flag) {
        $scope.allCheck = true;
      }
    } else {
      $scope.allCheck = false;
      angular.forEach(oneShop.shopGoods, function (innerItem) {
        innerItem.checked = false;
      });
    }
  };
  /*全部商铺商品选择*/
  $scope.allChecked = function () {
    if ($scope.allCheck) {
      angular.forEach($scope.allShops, function (oneShop) {
        oneShop.checked = true;
        angular.forEach(oneShop.shopGoods, function (innerItem) {
          innerItem.checked = true;
        });
      });
    } else {
      angular.forEach($scope.allShops, function (oneShop) {
        oneShop.checked = false;
        angular.forEach(oneShop.shopGoods, function (innerItem) {
          innerItem.checked = false;
        });
      });
    }
  };
});    
</script>
 
七、angular1两种作用域绑定
angular1模块、控制器和作用域的两种绑定方式,模块不必通过ng-app关联到HTML的标签上,也不必通过angular.bootstrap()关联到HTML的标签上!
1、正常绑定 
<!DOCTYPE html>
<html ng-app="app">
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<script src="http://apps.bdimg.com/libs/angular.js/1.4.6/angular.min.js"></script>
<body>
  <div>
    <div ng-controller="myCtrl">
      {{ hello }}
    </div>
  </div>
  <script type="text/javascript">
    var module = angular.module("app", []);
    module.controller("myCtrl", function ($scope) {
      //app模块的控制器(myCtrl)将作用域的hello 赋值为 "a Angular app"
      $scope.hello = "a Angular app";
    });
  </script>
</body>
</html>
2、异常绑定
<!DOCTYPE html>
<html>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<script src="http://apps.bdimg.com/libs/angular.js/1.4.6/angular.min.js"></script>
<body>
  <div id="app1">
    <div ng-controller="myCtrl">
      {{ hello }}
    </div>
  </div>
  <div id="app2">
    <div ng-controller="myCtrl">
      {{ hello }}
    </div>
  </div>
  <script type="text/javascript">
    var module1 = angular.module("test1", []);
    module1.controller("myCtrl", function ($scope) {
      //test1模块的控制器(myCtrl)将作用域的hello 赋值为 "a Angular app"
      $scope.hello = "a Angular app";
    });
    var module2 = angular.module("test2", []);
    module2.controller("myCtrl", function ($scope) {
      //test2模块的控制器(myCtrl)将作用域的hello 赋值为 "a Angular app"
      $scope.hello = " another Angular app";
    });
    /*此方法用于手动加载angularjs模板*/
    angular.bootstrap(document.getElementById("app1"), ['test1']);
    /*此方法用于手动加载angularjs模板*/
    angular.bootstrap(document.getElementById("app2"), ['test2']);
  </script>
</body>
</html>